diff --git a/.github/CODEOWNERS b/.github/CODEOWNERS deleted file mode 100644 index 3802c718..00000000 --- a/.github/CODEOWNERS +++ /dev/null @@ -1 +0,0 @@ -* @Nimond @Opti213 @YunusovSamat @lxmnk @vonabarak diff --git a/.github/workflows/deploy-docs-gh-pages.yml b/.github/workflows/deploy-docs-gh-pages.yml deleted file mode 100644 index 7a6b1e07..00000000 --- a/.github/workflows/deploy-docs-gh-pages.yml +++ /dev/null @@ -1,49 +0,0 @@ -name: Build and Deploy to GitHub Pages - -on: - push: - branches: - - master - -jobs: - build: - runs-on: ubuntu-18.04 - steps: - - uses: actions/checkout@v2 - - - name: Set up Python - uses: actions/setup-python@v1 - with: - python-version: 3.8 - - - name: Install poetry - run: | - pip install poetry==1.0 - poetry config virtualenvs.in-project true - - - name: Set up cache - uses: actions/cache@v1 - id: cache - with: - path: .venv - key: venv-${{ runner.os }}-py-${{ matrix.python-version }}-poetry-${{ hashFiles('poetry.lock') }} - - - name: Ensure cache is healthy - if: steps.cache.outputs.cache-hit == 'true' - shell: bash - run: poetry run pip --version >/dev/null 2>&1 || rm -rf .venv - - - name: Install dependencies - shell: bash - run: poetry install --extras tests - - - name: Build MkDocs for GitHub Pages - run: | - poetry run nox -s build-docs - - - name: Deploy to GitHub Pages - uses: JamesIves/github-pages-deploy-action@releases/v3 - with: - ACCESS_TOKEN: ${{ secrets.GH_PAGES_TOKEN }} - BRANCH: gh-pages - FOLDER: site \ No newline at end of file diff --git a/.github/workflows/deploy-docs-netlify.yml b/.github/workflows/deploy-docs-netlify.yml index 613ee0ba..4db3f0ab 100644 --- a/.github/workflows/deploy-docs-netlify.yml +++ b/.github/workflows/deploy-docs-netlify.yml @@ -1,13 +1,7 @@ name: Build and Deploy to Netlify on: - push: - branches: - - master - pull_request: - types: - - opened - - synchronize + push env: SITE_URL: https://pybotx.netlify.com @@ -23,31 +17,13 @@ jobs: with: python-version: 3.8 - - name: Install poetry - run: | - pip install poetry==1.0 - poetry config virtualenvs.in-project true - - - name: Set up cache - uses: actions/cache@v1 - id: cache - with: - path: .venv - key: venv-${{ runner.os }}-py-${{ matrix.python-version }}-poetry-${{ hashFiles('poetry.lock') }} - - - name: Ensure cache is healthy - if: steps.cache.outputs.cache-hit == 'true' - shell: bash - run: poetry run pip --version >/dev/null 2>&1 || rm -rf .venv - - - name: Install dependencies - shell: bash - run: poetry install --extras tests + - name: Setup dependencies + uses: ExpressApp/github-actions-poetry@v0.1 - name: Build MkDocs for Netlify run: | - poetry run nox -s build-docs - + source .venv/bin/activate + mkdocs build - name: Deploy to Netlify uses: nwtgck/actions-netlify@v1 with: @@ -56,4 +32,4 @@ jobs: github-token: ${{ secrets.GITHUB_TOKEN }} env: NETLIFY_AUTH_TOKEN: ${{ secrets.NETLIFY_AUTH_TOKEN }} - NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }} \ No newline at end of file + NETLIFY_SITE_ID: ${{ secrets.NETLIFY_SITE_ID }} diff --git a/.github/workflows/python_app.yml b/.github/workflows/python_app.yml new file mode 100644 index 00000000..3a61c2a7 --- /dev/null +++ b/.github/workflows/python_app.yml @@ -0,0 +1,54 @@ +name: Python application +on: push +jobs: + + test: + name: Test + runs-on: ubuntu-20.04 + strategy: + matrix: + python-version: ["3.8", "3.9", "3.10"] + + steps: + - name: Setup dependencies + uses: ExpressApp/github-actions-poetry@v0.2 + with: + python-version: ${{ matrix.python-version }} + poetry-version: "1.1.12" + + - name: Run tests + env: + BOT_CREDENTIALS: ${{ secrets.END_TO_END_TESTS_BOT_CREDENTIALS }} + run: | + poetry run ./scripts/test --cov-report=xml + + - name: Upload coverage to Codecov + uses: codecov/codecov-action@v2 + with: + fail_ci_if_error: true + files: ./coverage.xml + flags: unittests + + lint: + name: Lint + runs-on: ubuntu-20.04 + + steps: + - name: Setup dependencies + uses: ExpressApp/github-actions-poetry@v0.2 + + - name: Run linters + run: | + poetry run ./scripts/lint + + docs-lint: + name: Docs lint + runs-on: ubuntu-20.04 + + steps: + - name: Setup dependencies + uses: ExpressApp/github-actions-poetry@v0.2 + + - name: Run linters + run: | + poetry run ./scripts/docs-lint diff --git a/.github/workflows/styles.yml b/.github/workflows/styles.yml deleted file mode 100644 index ac0807cf..00000000 --- a/.github/workflows/styles.yml +++ /dev/null @@ -1,43 +0,0 @@ -name: Styles - -on: push - -jobs: - styles: - name: Styles - runs-on: ubuntu-18.04 - strategy: - matrix: - python-version: [3.8] - steps: - - uses: actions/checkout@v1 - - - name: Set up Python - uses: actions/setup-python@v1 - with: - python-version: ${{ matrix.python-version }} - - - name: Install poetry - run: | - pip install poetry==1.0 - poetry config virtualenvs.in-project true - - - name: Set up cache - uses: actions/cache@v1 - id: cache - with: - path: .venv - key: venv-${{ runner.os }}-py-${{ matrix.python-version }}-poetry-${{ hashFiles('poetry.lock') }} - - - name: Ensure cache is healthy - if: steps.cache.outputs.cache-hit == 'true' - shell: bash - run: poetry run pip --version >/dev/null 2>&1 || rm -rf .venv - - - name: Install dependencies - shell: bash - run: poetry install --extras tests - - - name: Run linters - run: | - poetry run nox -s lint diff --git a/.github/workflows/tests.yml b/.github/workflows/tests.yml deleted file mode 100644 index 577c321e..00000000 --- a/.github/workflows/tests.yml +++ /dev/null @@ -1,48 +0,0 @@ -name: Tests - -on: push - -jobs: - tests: - name: Tests - runs-on: ubuntu-18.04 - strategy: - matrix: - python-version: [3.7, 3.8, 3.9] - steps: - - uses: actions/checkout@master - - - name: Set up Python - uses: actions/setup-python@v1 - with: - python-version: ${{ matrix.python-version }} - - - name: Install poetry - run: | - pip install poetry==1.0 - poetry config virtualenvs.in-project true - - - name: Set up cache - uses: actions/cache@v1 - id: cache - with: - path: .venv - key: venv-${{ runner.os }}-py-${{ matrix.python-version }}-poetry-${{ hashFiles('poetry.lock') }} - - - name: Ensure cache is healthy - if: steps.cache.outputs.cache-hit == 'true' - shell: bash - run: poetry run pip --version >/dev/null 2>&1 || rm -rf .venv - - - name: Install dependencies - shell: bash - run: poetry install --extras tests - - - name: Run tests - run: | - poetry run nox -s test - - - uses: codecov/codecov-action@v1 - with: - file: ./coverage.xml - fail_ci_if_error: true diff --git a/.gitignore b/.gitignore index 1a1155c1..d91383cb 100644 --- a/.gitignore +++ b/.gitignore @@ -1,111 +1,8 @@ -# Byte-compiled / optimized / DLL files -__pycache__/ -*.py[cod] -*$py.class - -# C extensions -*.so - -# Distribution / packaging -.Python -build/ -develop-eggs/ -dist/ -downloads/ -eggs/ -.eggs/ -lib/ -lib64/ -parts/ -sdist/ -var/ -wheels/ -*.egg-info/ -.installed.cfg -*.egg -MANIFEST - -# PyInstaller -# Usually these files are written by a python script from a template -# before PyInstaller builds the exe, so as to inject date/other infos into it. -*.manifest -*.spec - -# Installer logs -pip-log.txt -pip-delete-this-directory.txt - -# Unit test / coverage reports -htmlcov/ -.tox/ .coverage -.coverage.* -.cache -nosetests.xml -coverage.xml -*.cover -.hypothesis/ -.pytest_cache/ - -# Translations -*.mo -*.pot - -# Django stuff: -*.log -local_settings.py -db.sqlite3 - -# Flask stuff: -instance/ -.webassets-cache - -# Scrapy stuff: -.scrapy - -# Sphinx documentation -docs/_build/ - -# PyBuilder -target/ - -# Jupyter Notebook -.ipynb_checkpoints - -# pyenv -.python-version - -# celery beat schedule file -celerybeat-schedule - -# SageMath parsed files -*.sage.py - -# Environments .env -.venv -env/ +.venv/ venv/ -ENV/ -env.bak/ -venv.bak/ - -# Spyder project settings -.spyderproject -.spyproject - -# Rope project settings -.ropeproject - -# mkdocs documentation -/site - -# mypy -.mypy_cache/ - +__pycache__ +htmlcov +site .idea/ -.vscode/ - -static/ - -**/.DS_Store diff --git a/README.md b/README.md index 4cac9024..ac2a8dd7 100644 --- a/README.md +++ b/README.md @@ -1,126 +1,37 @@ -

pybotx

-

- A little python framework for building bots for eXpress messenger. -

-

- - Tests - - - Styles - - - Coverage - - - Code Style - - - Package version - - - License - -

+# pybotx +*A python library for building bots and smartapps for eXpress messenger.* ---- +[![PyPI version](https://badge.fury.io/py/botx.svg)](https://badge.fury.io/py/botx) +![PyPI - Python Version](https://img.shields.io/pypi/pyversions/botx) +[![Coverage](https://codecov.io/gh/ExpressApp/pybotx/branch/master/graph/badge.svg)](https://codecov.io/gh/ExpressApp/pybotx/branch/master) +[![Code style](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/ambv/black) -# Introduction -`pybotx` is a framework for building bots for eXpress providing a mechanism for simple -integration with your favourite web frameworks. +## Features -Main features: +* Designed to be easy to use +* Simple integration with async web-frameworks +* Support middlewares for command, command-collector and bot +* 100% test coverage +* 100% type annotated codebase - * Simple integration with your web apps. - * Asynchronous API with synchronous as a fallback option. - * 100% test coverage. - * 100% type annotated codebase. +## Documentation -**NOTE**: *This library is under active development and its API may be unstable. Please lock the version you are using at the minor update level. For example, like this in `poetry`.* +Documentation will be here: +For now, pls contact eXpress team, we'll help you. -```toml -[tool.poetry.dependencies] -botx = "^0.15.0" -``` - ---- - -## Requirements - -Python 3.7+ +**Note:** Available only in Russian language -`pybotx` use the following libraries: - -* pydantic for the data parts. -* httpx for making HTTP calls to BotX API. -* loguru for beautiful and powerful logs. -* **Optional**. Starlette for tests. ## Installation -```bash -$ pip install botx -``` - -Or if you are going to write tests: - -```bash -$ pip install botx[tests] -``` - -You will also need a web framework to create bots as the current BotX API only works with webhooks. -This documentation will use FastAPI for the examples bellow. -```bash -$ pip install fastapi uvicorn -``` - -## Example -Let's create a simple echo bot. +Install pybotx using `pip`: -* Create a file `main.py` with following content: - -```python3 -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED -from uuid import UUID - - -bot_accounts=[ - BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id")) -] -bot = Bot(bot_accounts=bot_accounts) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) -``` - -* Deploy a bot on your server using uvicorn and set the url for the webhook in Express. ```bash -$ uvicorn main:app --host=0.0.0.0 +pip install git+https://github.com/ExpressApp/pybotx.git ``` -This bot will send back every your message. - -## License - -This project is licensed under the terms of the MIT license. +**Note:** This project is under active development (`0.y.z`) and its API may be +unstable. diff --git a/botx/__init__.py b/botx/__init__.py index 0bdf1196..ecd2d420 100644 --- a/botx/__init__.py +++ b/botx/__init__.py @@ -1,157 +1,173 @@ -"""A little python framework for building bots for Express.""" - -from loguru import logger - -from botx.bots.bots import Bot -from botx.clients.clients.async_client import AsyncClient -from botx.clients.clients.sync_client import Client -from botx.clients.types.message_payload import InternalBotNotificationPayload -from botx.collecting.collectors.collector import Collector -from botx.dependencies.injection_params import Depends -from botx.exceptions import BotXAPIError, DependencyFailure, TokenError, UnknownBotError -from botx.models.attachments import ( - AttachList, - Attachment, - Contact, - Document, - Image, - Link, - Location, - Video, - Voice, +from botx.bot.api.exceptions import UnsupportedBotAPIVersionError +from botx.bot.api.responses.bot_disabled import ( + BotAPIBotDisabledResponse, + build_bot_disabled_response, +) +from botx.bot.api.responses.command_accepted import build_command_accepted_response +from botx.bot.bot import Bot +from botx.bot.exceptions import ( + AnswerDestinationLookupError, + BotShuttingDownError, + BotXMethodCallbackNotFoundError, + UnknownBotAccountError, +) +from botx.bot.handler import IncomingMessageHandlerFunc, Middleware +from botx.bot.handler_collector import HandlerCollector +from botx.bot.testing import lifespan_wrapper +from botx.client.exceptions.callbacks import ( + BotXMethodFailedCallbackReceivedError, + CallbackNotReceivedError, +) +from botx.client.exceptions.chats import ( + CantUpdatePersonalChatError, + ChatCreationError, + ChatCreationProhibitedError, + InvalidUsersListError, +) +from botx.client.exceptions.common import ( + ChatNotFoundError, + InvalidBotAccountError, + PermissionDeniedError, + RateLimitReachedError, +) +from botx.client.exceptions.event import EventNotFoundError +from botx.client.exceptions.files import FileDeletedError, FileMetadataNotFound +from botx.client.exceptions.http import ( + InvalidBotXResponsePayloadError, + InvalidBotXStatusCodeError, +) +from botx.client.exceptions.notifications import ( + BotIsNotChatMemberError, + FinalRecipientsListEmptyError, + StealthModeDisabledError, ) -from botx.models.buttons import BubbleElement, KeyboardElement -from botx.models.credentials import BotXCredentials -from botx.models.entities import ( - ChatMention, - Entity, - EntityList, - Forward, - Mention, - Reply, - UserMention, +from botx.client.exceptions.users import UserNotFoundError +from botx.client.stickers_api.exceptions import ( + InvalidEmojiError, + InvalidImageError, + StickerPackOrStickerNotFoundError, ) +from botx.logger import logger +from botx.models.async_files import Document, File, Image, Video, Voice +from botx.models.attachments import OutgoingAttachment +from botx.models.bot_account import BotAccount, BotAccountWithSecret +from botx.models.bot_sender import BotSender +from botx.models.chats import Chat, ChatInfo, ChatInfoMember, ChatListItem from botx.models.enums import ( + AttachmentTypes, ChatTypes, - CommandTypes, - EntityTypes, + ClientPlatforms, MentionTypes, - Statuses, - SystemEvents, UserKinds, ) -from botx.models.errors import BotDisabledErrorData, BotDisabledResponse -from botx.models.events import ChatCreatedEvent, InternalBotNotificationEvent -from botx.models.files import File -from botx.models.menu import Status -from botx.models.messages.incoming_message import IncomingMessage -from botx.models.messages.message import Message -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.markup import MessageMarkup -from botx.models.messages.sending.message import SendingMessage -from botx.models.messages.sending.options import MessageOptions, NotificationOptions -from botx.models.messages.sending.payload import MessagePayload, UpdatePayload -from botx.models.smartapps import SendingSmartAppEvent, SendingSmartAppNotification -from botx.models.status import StatusRecipient -from botx.models.stickers import ( - Pagination, - Sticker, - StickerFromPack, - StickerPack, - StickerPackList, - StickerPackPreview, +from botx.models.message.edit_message import EditMessage +from botx.models.message.forward import Forward +from botx.models.message.incoming_message import IncomingMessage, UserDevice, UserSender +from botx.models.message.markup import BubbleMarkup, Button, KeyboardMarkup +from botx.models.message.mentions import Mention, MentionList +from botx.models.message.message_status import MessageStatus +from botx.models.message.outgoing_message import OutgoingMessage +from botx.models.message.reply import Reply +from botx.models.message.reply_message import ReplyMessage +from botx.models.method_callbacks import BotAPIMethodFailedCallback +from botx.models.status import BotMenu, StatusRecipient +from botx.models.stickers import Sticker, StickerPack +from botx.models.system_events.added_to_chat import AddedToChatEvent +from botx.models.system_events.chat_created import ChatCreatedEvent, ChatCreatedMember +from botx.models.system_events.cts_login import CTSLoginEvent +from botx.models.system_events.cts_logout import CTSLogoutEvent +from botx.models.system_events.deleted_from_chat import DeletedFromChatEvent +from botx.models.system_events.internal_bot_notification import ( + InternalBotNotificationEvent, ) -from botx.testing.building.builder import MessageBuilder - -try: - from botx.testing.testing_client.client import TestClient # noqa: WPS433 -except ImportError: - TestClient = None # type: ignore # noqa: WPS440 +from botx.models.system_events.left_from_chat import LeftFromChatEvent +from botx.models.system_events.smartapp_event import SmartAppEvent +from botx.models.users import UserFromSearch __all__ = ( - # bots + "AddedToChatEvent", + "AnswerDestinationLookupError", + "AttachmentTypes", "Bot", - # collecting - "Collector", - # clients - "AsyncClient", - "Client", - # exceptions - "BotXAPIError", - "DependencyFailure", - "UnknownBotError", - "TokenError", - # DI - "Depends", - # models - # markup - "BubbleElement", - "KeyboardElement", - # credentials - "BotXCredentials", - # enums - "Statuses", - "UserKinds", - "ChatTypes", - "CommandTypes", - "SystemEvents", - "EntityTypes", - "MentionTypes", - # errors - "BotDisabledErrorData", - "BotDisabledResponse", - # events + "BotAccountWithSecret", + "BotAPIBotDisabledResponse", + "BotAPIMethodFailedCallback", + "BotIsNotChatMemberError", + "BotMenu", + "BotAccount", + "BotSender", + "BotShuttingDownError", + "BotXMethodCallbackNotFoundError", + "BotXMethodFailedCallbackReceivedError", + "BubbleMarkup", + "Button", + "CallbackNotReceivedError", + "CantUpdatePersonalChatError", + "Chat", "ChatCreatedEvent", - "InternalBotNotificationEvent", - # files + "ChatCreatedMember", + "ChatCreationError", + "ChatCreationProhibitedError", + "ChatInfo", + "ChatInfoMember", + "ChatListItem", + "ChatNotFoundError", + "ChatTypes", + "ClientPlatforms", + "CTSLoginEvent", + "CTSLogoutEvent", + "DeletedFromChatEvent", + "Document", + "EditMessage", + "MessageStatus", "File", - # attachments + "FileDeletedError", + "FileMetadataNotFound", + "FinalRecipientsListEmptyError", + "Forward", + "HandlerCollector", "Image", - "Video", - "Document", - "Voice", - "Location", - "Contact", - "Link", - "Attachment", - "AttachList", - # mentions - "Mention", - "ChatMention", - "UserMention", - # status - "Status", - "StatusRecipient", - # messages - # handler message - "Message", - # incoming "IncomingMessage", - "Entity", - "EntityList", - "Forward", + "IncomingMessageHandlerFunc", + "InternalBotNotificationEvent", + "InvalidBotAccountError", + "InvalidBotXResponsePayloadError", + "InvalidBotXStatusCodeError", + "InvalidEmojiError", + "InvalidImageError", + "InvalidUsersListError", + "EventNotFoundError", + "KeyboardMarkup", + "LeftFromChatEvent", + "Mention", + "MentionList", + "MentionTypes", + "Middleware", + "OutgoingAttachment", + "OutgoingMessage", + "PermissionDeniedError", + "RateLimitReachedError", "Reply", - # sending - "SendingCredentials", - "SendingMessage", - "MessageMarkup", - "MessageOptions", - "NotificationOptions", - "MessagePayload", - "UpdatePayload", - "InternalBotNotificationPayload", - # Stickers - "Pagination", + "ReplyMessage", + "SmartAppEvent", + "SmartAppEvent", + "StatusRecipient", + "StealthModeDisabledError", "Sticker", "StickerPack", - "StickerPackList", - "StickerFromPack", - "StickerPackPreview", - "SendingSmartAppEvent", - "SendingSmartAppNotification", - # testing - "TestClient", - "MessageBuilder", + "StickerPackOrStickerNotFoundError", + "UnknownBotAccountError", + "UnsupportedBotAPIVersionError", + "UserDevice", + "UserFromSearch", + "UserKinds", + "UserNotFoundError", + "UserSender", + "Video", + "Voice", + "build_bot_disabled_response", + "build_command_accepted_response", + "lifespan_wrapper", ) logger.disable("botx") diff --git a/botx/async_buffer.py b/botx/async_buffer.py new file mode 100644 index 00000000..d5e8a8a5 --- /dev/null +++ b/botx/async_buffer.py @@ -0,0 +1,33 @@ +import os +from typing import Optional + +try: + from typing import Protocol +except ImportError: + from typing_extensions import Protocol # type: ignore # noqa: WPS440 + + +class AsyncBufferBase(Protocol): + async def seek(self, cursor: int, whence: int = os.SEEK_SET) -> int: + ... # noqa: WPS428 + + async def tell(self) -> int: + ... # noqa: WPS428 + + +class AsyncBufferWritable(AsyncBufferBase): + async def write(self, content: bytes) -> int: + ... # noqa: WPS428 + + +class AsyncBufferReadable(AsyncBufferBase): + async def read(self, bytes_to_read: Optional[int] = None) -> bytes: + ... # noqa: WPS428 + + +async def get_file_size(async_buffer: AsyncBufferReadable) -> int: + await async_buffer.seek(0, os.SEEK_END) + file_size = await async_buffer.tell() + await async_buffer.seek(0) + + return file_size diff --git a/docs/src/__init__.py b/botx/bot/__init__.py similarity index 100% rename from docs/src/__init__.py rename to botx/bot/__init__.py diff --git a/docs/src/development/__init__.py b/botx/bot/api/__init__.py similarity index 100% rename from docs/src/development/__init__.py rename to botx/bot/api/__init__.py diff --git a/botx/bot/api/exceptions.py b/botx/bot/api/exceptions.py new file mode 100644 index 00000000..47cd0a5d --- /dev/null +++ b/botx/bot/api/exceptions.py @@ -0,0 +1,10 @@ +from botx.constants import BOT_API_VERSION + + +class UnsupportedBotAPIVersionError(Exception): + def __init__(self, version: int) -> None: + self.version = version + self.message = ( + f"Unsupported Bot API version: `{version}`, expected `{BOT_API_VERSION}`" + ) + super().__init__(self.message) diff --git a/docs/src/development/collector/__init__.py b/botx/bot/api/responses/__init__.py similarity index 100% rename from docs/src/development/collector/__init__.py rename to botx/bot/api/responses/__init__.py diff --git a/botx/bot/api/responses/bot_disabled.py b/botx/bot/api/responses/bot_disabled.py new file mode 100644 index 00000000..c8a2f3af --- /dev/null +++ b/botx/bot/api/responses/bot_disabled.py @@ -0,0 +1,41 @@ +from dataclasses import asdict, dataclass, field +from typing import Any, Dict, List, Literal + + +@dataclass +class BotAPIBotDisabledErrorData: + status_message: str + + +@dataclass +class BotAPIBotDisabledResponse: + """Disabled bot response model. + + Only `.error_data.status_message` attribute will be displayed to + user. Other attributes will be visible only in BotX logs. + """ + + error_data: BotAPIBotDisabledErrorData + errors: List[str] = field(default_factory=list) + reason: Literal["bot_disabled"] = "bot_disabled" + + +def build_bot_disabled_response(status_message: str) -> Dict[str, Any]: + """Build bot disabled response for BotX. + + It should be send if the bot can't process the command. + + If you would like to build complex response, see + [BotAPIBotDisabledResponse] + [botx.bot.api.commands.bot_disabled_response.BotAPIBotDisabledResponse]. + + :param status_message: Status message. + + :return: Built bot disabled response. + """ + + response = BotAPIBotDisabledResponse( + error_data=BotAPIBotDisabledErrorData(status_message=status_message), + ) + + return asdict(response) diff --git a/botx/bot/api/responses/command_accepted.py b/botx/bot/api/responses/command_accepted.py new file mode 100644 index 00000000..a0713734 --- /dev/null +++ b/botx/bot/api/responses/command_accepted.py @@ -0,0 +1,12 @@ +from typing import Any, Dict + + +def build_command_accepted_response() -> Dict[str, Any]: + """Build accepted response for BotX. + + It should be sent if the bot started processing a command. + + :return: Built accepted response. + """ + + return {"result": "accepted"} diff --git a/botx/bot/bot.py b/botx/bot/bot.py new file mode 100644 index 00000000..9d3d8c56 --- /dev/null +++ b/botx/bot/bot.py @@ -0,0 +1,1466 @@ +from types import SimpleNamespace +from typing import Any, AsyncIterable, Dict, Iterator, List, Optional, Sequence, Union +from uuid import UUID + +import httpx +from pydantic import ValidationError, parse_obj_as + +from botx.async_buffer import AsyncBufferReadable, AsyncBufferWritable +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.bot.callbacks_manager import CallbacksManager +from botx.bot.contextvars import bot_id_var, chat_id_var +from botx.bot.exceptions import AnswerDestinationLookupError +from botx.bot.handler import Middleware +from botx.bot.handler_collector import HandlerCollector +from botx.bot.middlewares.exception_middleware import ExceptionHandlersDict +from botx.client.chats_api.add_admin import ( + AddAdminMethod, + BotXAPIAddAdminRequestPayload, +) +from botx.client.chats_api.add_user import AddUserMethod, BotXAPIAddUserRequestPayload +from botx.client.chats_api.chat_info import ( + BotXAPIChatInfoRequestPayload, + ChatInfoMethod, +) +from botx.client.chats_api.create_chat import ( + BotXAPICreateChatRequestPayload, + CreateChatMethod, +) +from botx.client.chats_api.disable_stealth import ( + BotXAPIDisableStealthRequestPayload, + DisableStealthMethod, +) +from botx.client.chats_api.list_chats import ListChatsMethod +from botx.client.chats_api.pin_message import ( + BotXAPIPinMessageRequestPayload, + PinMessageMethod, +) +from botx.client.chats_api.remove_user import ( + BotXAPIRemoveUserRequestPayload, + RemoveUserMethod, +) +from botx.client.chats_api.set_stealth import ( + BotXAPISetStealthRequestPayload, + SetStealthMethod, +) +from botx.client.chats_api.unpin_message import ( + BotXAPIUnpinMessageRequestPayload, + UnpinMessageMethod, +) +from botx.client.events_api.edit_event import ( + BotXAPIEditEventRequestPayload, + EditEventMethod, +) +from botx.client.events_api.message_status_event import ( + BotXAPIMessageStatusRequestPayload, + MessageStatusMethod, +) +from botx.client.events_api.reply_event import ( + BotXAPIReplyEventRequestPayload, + ReplyEventMethod, +) +from botx.client.events_api.stop_typing_event import ( + BotXAPIStopTypingEventRequestPayload, + StopTypingEventMethod, +) +from botx.client.events_api.typing_event import ( + BotXAPITypingEventRequestPayload, + TypingEventMethod, +) +from botx.client.exceptions.common import InvalidBotAccountError +from botx.client.files_api.download_file import ( + BotXAPIDownloadFileRequestPayload, + DownloadFileMethod, +) +from botx.client.files_api.upload_file import ( + BotXAPIUploadFileRequestPayload, + UploadFileMethod, +) +from botx.client.get_token import get_token +from botx.client.notifications_api.direct_notification import ( + BotXAPIDirectNotificationRequestPayload, + DirectNotificationMethod, +) +from botx.client.notifications_api.internal_bot_notification import ( + BotXAPIInternalBotNotificationRequestPayload, + InternalBotNotificationMethod, +) +from botx.client.smartapps_api.smartapp_event import ( + BotXAPISmartAppEventRequestPayload, + SmartAppEventMethod, +) +from botx.client.smartapps_api.smartapp_notification import ( + BotXAPISmartAppNotificationRequestPayload, + SmartAppNotificationMethod, +) +from botx.client.stickers_api.add_sticker import ( + AddStickerMethod, + BotXAPIAddStickerRequestPayload, +) +from botx.client.stickers_api.create_sticker_pack import ( + BotXAPICreateStickerPackRequestPayload, + CreateStickerPackMethod, +) +from botx.client.stickers_api.delete_sticker import ( + BotXAPIDeleteStickerRequestPayload, + DeleteStickerMethod, +) +from botx.client.stickers_api.delete_sticker_pack import ( + BotXAPIDeleteStickerPackRequestPayload, + DeleteStickerPackMethod, +) +from botx.client.stickers_api.edit_sticker_pack import ( + BotXAPIEditStickerPackRequestPayload, + EditStickerPackMethod, +) +from botx.client.stickers_api.get_sticker import ( + BotXAPIGetStickerRequestPayload, + GetStickerMethod, +) +from botx.client.stickers_api.get_sticker_pack import ( + BotXAPIGetStickerPackRequestPayload, + GetStickerPackMethod, +) +from botx.client.stickers_api.get_sticker_packs import ( + BotXAPIGetStickerPacksRequestPayload, + GetStickerPacksMethod, +) +from botx.client.users_api.search_user_by_email import ( + BotXAPISearchUserByEmailRequestPayload, + SearchUserByEmailMethod, +) +from botx.client.users_api.search_user_by_huid import ( + BotXAPISearchUserByHUIDRequestPayload, + SearchUserByHUIDMethod, +) +from botx.client.users_api.search_user_by_login import ( + BotXAPISearchUserByLoginRequestPayload, + SearchUserByLoginMethod, +) +from botx.constants import STICKER_PACKS_PER_PAGE +from botx.converters import optional_sequence_to_list +from botx.image_validators import ( + ensure_file_content_is_png, + ensure_sticker_image_size_valid, +) +from botx.logger import logger, pformat_jsonable_obj, trim_file_data_in_incoming_json +from botx.missing import Missing, MissingOptional, Undefined, not_undefined +from botx.models.async_files import File +from botx.models.attachments import IncomingFileAttachment, OutgoingAttachment +from botx.models.bot_account import BotAccount, BotAccountWithSecret +from botx.models.chats import ChatInfo, ChatListItem +from botx.models.commands import BotAPICommand, BotCommand +from botx.models.enums import ChatTypes +from botx.models.message.edit_message import EditMessage +from botx.models.message.markup import BubbleMarkup, KeyboardMarkup +from botx.models.message.message_status import MessageStatus +from botx.models.message.outgoing_message import OutgoingMessage +from botx.models.message.reply_message import ReplyMessage +from botx.models.method_callbacks import BotXMethodCallback +from botx.models.status import ( + BotAPIStatusRecipient, + BotMenu, + StatusRecipient, + build_bot_status_response, +) +from botx.models.stickers import Sticker, StickerPack, StickerPackFromList +from botx.models.users import UserFromSearch + +MissingOptionalAttachment = MissingOptional[ + Union[IncomingFileAttachment, OutgoingAttachment] +] + + +class Bot: + def __init__( + self, + *, + collectors: Sequence[HandlerCollector], + bot_accounts: Sequence[BotAccountWithSecret], + middlewares: Optional[Sequence[Middleware]] = None, + httpx_client: Optional[httpx.AsyncClient] = None, + exception_handlers: Optional[ExceptionHandlersDict] = None, + default_callback_timeout: Optional[int] = None, + ) -> None: + if not collectors: + logger.warning("Bot has no connected collectors") + if not bot_accounts: + logger.warning("Bot has no bot accounts") + + self.state: SimpleNamespace = SimpleNamespace() + + self.default_callback_timeout = default_callback_timeout + + middlewares = optional_sequence_to_list(middlewares) + + self._handler_collector = self._build_main_collector( + collectors, + middlewares, + exception_handlers, + ) + + self._bot_accounts_storage = BotAccountsStorage(list(bot_accounts)) + self._httpx_client = httpx_client or httpx.AsyncClient() + + self._callback_manager = CallbacksManager() + + def async_execute_raw_bot_command(self, raw_bot_command: Dict[str, Any]) -> None: + try: + bot_api_command: BotAPICommand = parse_obj_as( + # Same ignore as in pydantic + BotAPICommand, # type: ignore[arg-type] + raw_bot_command, + ) + except ValidationError as validation_exc: + raise ValueError("Bot command validation error") from validation_exc + + logger.opt(lazy=True).debug( + "Got command: {command}", + command=lambda: pformat_jsonable_obj( + trim_file_data_in_incoming_json(raw_bot_command), + ), + ) + + bot_command = bot_api_command.to_domain(raw_bot_command) + self.async_execute_bot_command(bot_command) + + def async_execute_bot_command(self, bot_command: BotCommand) -> None: + # raise UnknownBotAccountError if no bot account with this bot_id. + self._bot_accounts_storage.ensure_bot_id_exists(bot_command.bot.id) + + self._handler_collector.async_handle_bot_command(self, bot_command) + + async def raw_get_status(self, query_params: Dict[str, str]) -> Dict[str, Any]: + logger.opt(lazy=True).debug( + "Got status: {status}", + status=lambda: pformat_jsonable_obj(query_params), + ) + + try: + bot_api_status_recipient = BotAPIStatusRecipient.parse_obj(query_params) + except ValidationError as exc: + raise ValueError("Status request validation error") from exc + + status_recipient = bot_api_status_recipient.to_domain() + + bot_menu = await self.get_status(status_recipient) + return build_bot_status_response(bot_menu) + + async def get_status(self, status_recipient: StatusRecipient) -> BotMenu: + # raise UnknownBotAccountError if no bot account with this bot_id. + self._bot_accounts_storage.ensure_bot_id_exists(status_recipient.bot_id) + + return await self._handler_collector.get_bot_menu(status_recipient, self) + + def set_raw_botx_method_result( + self, + raw_botx_method_result: Dict[str, Any], + ) -> None: + logger.debug("Got callback: {callback}", callback=raw_botx_method_result) + + callback: BotXMethodCallback = parse_obj_as( + # Same ignore as in pydantic + BotXMethodCallback, # type: ignore[arg-type] + raw_botx_method_result, + ) + + self._callback_manager.set_botx_method_callback_result(callback) + + async def wait_botx_method_callback( + self, + sync_id: UUID, + timeout: Optional[int], + ) -> BotXMethodCallback: + return await self._callback_manager.wait_botx_method_callback(sync_id, timeout) + + @property + def bot_accounts(self) -> Iterator[BotAccount]: + yield from self._bot_accounts_storage.iter_bot_accounts() + + async def startup(self) -> None: + for bot_account in self.bot_accounts: + try: + token = await self.get_token(bot_id=bot_account.id) + except (InvalidBotAccountError, httpx.HTTPError): + logger.warning( + "Can't get token for bot account: " + f"host - {bot_account.host}, bot_id - {bot_account.id}", + ) + continue + + self._bot_accounts_storage.set_token(bot_account.id, token) + + async def shutdown(self) -> None: + self._callback_manager.stop_callbacks_waiting() + await self._handler_collector.wait_active_tasks() + await self._httpx_client.aclose() + + # - Bots API - + async def get_token( + self, + *, + bot_id: UUID, + ) -> str: + """Get bot auth token. + + :param bot_id: Bot which should perform the request. + + :return: Auth token. + """ + + return await get_token(bot_id, self._httpx_client, self._bot_accounts_storage) + + # - Notifications API - + async def answer_message( + self, + body: str, + *, + metadata: Missing[Dict[str, Any]] = Undefined, + bubbles: Missing[BubbleMarkup] = Undefined, + keyboard: Missing[KeyboardMarkup] = Undefined, + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined, + recipients: Missing[List[UUID]] = Undefined, + silent_response: Missing[bool] = Undefined, + markup_auto_adjust: Missing[bool] = Undefined, + stealth_mode: Missing[bool] = Undefined, + send_push: Missing[bool] = Undefined, + ignore_mute: Missing[bool] = Undefined, + wait_callback: bool = True, + callback_timeout: MissingOptional[int] = Undefined, + ) -> UUID: + """Answer to incoming message. + + Works just like `Bot.send`, but `bot_id` and `chat_id` are + taken from the incoming message. + + :param body: Message body. + :param metadata: Notification options. + :param bubbles: Bubbles (buttons attached to message) markup. + :param keyboard: Keyboard (buttons below message input) markup. + :param file: Attachment. + :param recipients: List of recipients, empty for all in chat. + :param silent_response: (BotX default: False) Exclude next user + messages from history. + :param markup_auto_adjust: (BotX default: False) Move button to next + row, if its text doesn't fit. + :param stealth_mode: (BotX default: False) Enable stealth mode. + :param send_push: (BotX default: True) Send push notification on + devices. + :param ignore_mute: (BotX default: False) Ignore mute or dnd (do not + disturb). + :param wait_callback: Block method call until callback received. + :param callback_timeout: Callback timeout in seconds (or `None` for + endless waiting). + + :raises AnswerDestinationLookupError: If you try to answer without + receiving incoming message. + + :return: Notification sync_id. + """ + + try: # noqa: WPS229 + bot_id = bot_id_var.get() + chat_id = chat_id_var.get() + except LookupError as exc: + raise AnswerDestinationLookupError from exc + + return await self.send_message( + bot_id=bot_id, + chat_id=chat_id, + body=body, + metadata=metadata, + bubbles=bubbles, + keyboard=keyboard, + file=file, + recipients=recipients, + silent_response=silent_response, + markup_auto_adjust=markup_auto_adjust, + stealth_mode=stealth_mode, + send_push=send_push, + ignore_mute=ignore_mute, + wait_callback=wait_callback, + callback_timeout=callback_timeout, + ) + + async def send( + self, + *, + message: OutgoingMessage, + wait_callback: bool = True, + callback_timeout: MissingOptional[int] = Undefined, + ) -> UUID: + """Send internal notification. + + :param message: Built outgoing message. + :param wait_callback: Wait for callback. + :param callback_timeout: Timeout for waiting for callback. + + :return: Notification sync_id. + """ + + return await self.send_message( + bot_id=message.bot_id, + chat_id=message.chat_id, + body=message.body, + metadata=message.metadata, + bubbles=message.bubbles, + keyboard=message.keyboard, + file=message.file, + recipients=message.recipients, + silent_response=message.silent_response, + markup_auto_adjust=message.markup_auto_adjust, + stealth_mode=message.stealth_mode, + send_push=message.send_push, + ignore_mute=message.ignore_mute, + wait_callback=wait_callback, + callback_timeout=callback_timeout, + ) + + async def send_message( + self, + *, + bot_id: UUID, + chat_id: UUID, + body: str, + metadata: Missing[Dict[str, Any]] = Undefined, + bubbles: Missing[BubbleMarkup] = Undefined, + keyboard: Missing[KeyboardMarkup] = Undefined, + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined, + silent_response: Missing[bool] = Undefined, + markup_auto_adjust: Missing[bool] = Undefined, + recipients: Missing[List[UUID]] = Undefined, + stealth_mode: Missing[bool] = Undefined, + send_push: Missing[bool] = Undefined, + ignore_mute: Missing[bool] = Undefined, + wait_callback: bool = True, + callback_timeout: MissingOptional[int] = Undefined, + ) -> UUID: + """Send message to chat. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param body: Message body. + :param metadata: Notification options. + :param bubbles: Bubbles (buttons attached to message) markup. + :param keyboard: Keyboard (buttons below message input) markup. + :param file: Attachment. + :param recipients: List of recipients, empty for all in chat. + :param silent_response: (BotX default: False) Exclude next user + messages from history. + :param markup_auto_adjust: (BotX default: False) Move button to next + row, if its text doesn't fit. + :param stealth_mode: (BotX default: False) Enable stealth mode. + :param send_push: (BotX default: True) Send push notification on + devices. + :param ignore_mute: (BotX default: False) Ignore mute or dnd (do not + disturb). + :param wait_callback: Block method call until callback received. + :param callback_timeout: Callback timeout in seconds (or `None` for + endless waiting). + + :return: Notification sync_id. + """ + + method = DirectNotificationMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + self._callback_manager, + ) + + payload = BotXAPIDirectNotificationRequestPayload.from_domain( + chat_id=chat_id, + body=body, + metadata=metadata, + bubbles=bubbles, + keyboard=keyboard, + file=file, + recipients=recipients, + silent_response=silent_response, + markup_auto_adjust=markup_auto_adjust, + stealth_mode=stealth_mode, + send_push=send_push, + ignore_mute=ignore_mute, + ) + botx_api_sync_id = await method.execute( + payload, + wait_callback, + not_undefined(callback_timeout, self.default_callback_timeout), + ) + + return botx_api_sync_id.to_domain() + + async def send_internal_bot_notification( + self, + *, + bot_id: UUID, + chat_id: UUID, + data: Dict[str, Any], + opts: Missing[Dict[str, Any]] = Undefined, + recipients: Missing[List[UUID]] = Undefined, + wait_callback: bool = True, + callback_timeout: MissingOptional[int] = Undefined, + ) -> UUID: + """Send internal notification. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param data: Notification payload. + :param opts: Notification options. + :param recipients: List of bot uuids, empty for all in chat. + :param wait_callback: Wait for callback. + :param callback_timeout: Timeout for waiting for callback. + + :return: Notification sync_id. + """ + + method = InternalBotNotificationMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + self._callback_manager, + ) + + payload = BotXAPIInternalBotNotificationRequestPayload.from_domain( + chat_id=chat_id, + data=data, + opts=opts, + recipients=recipients, + ) + botx_api_sync_id = await method.execute( + payload, + wait_callback, + not_undefined(callback_timeout, self.default_callback_timeout), + ) + + return botx_api_sync_id.to_domain() + + # - Events API - + async def edit( + self, + *, + message: EditMessage, + ) -> None: + """Edit message. + + :param message: Built outgoing edit message. + """ + + await self.edit_message( + bot_id=message.bot_id, + sync_id=message.sync_id, + body=message.body, + metadata=message.metadata, + bubbles=message.bubbles, + keyboard=message.keyboard, + file=message.file, + markup_auto_adjust=message.markup_auto_adjust, + ) + + async def edit_message( + self, + *, + bot_id: UUID, + sync_id: UUID, + body: Missing[str] = Undefined, + metadata: Missing[Dict[str, Any]] = Undefined, + bubbles: Missing[BubbleMarkup] = Undefined, + keyboard: Missing[KeyboardMarkup] = Undefined, + file: MissingOptionalAttachment = Undefined, + markup_auto_adjust: Missing[bool] = Undefined, + ) -> None: + """Edit message. + + :param bot_id: Bot which should perform the request. + :param sync_id: `sync_id` of message to update. + :param body: New message body. Skip to leave previous body or pass + empty string to clean it. + :param metadata: Notification options. Skip to leave previous metadata. + :param bubbles: Bubbles (buttons attached to message) markup. Skip to + leave previous bubbles. + :param keyboard: Keyboard (buttons below message input) markup. Skip + to leave previous keyboard. + :param file: Attachment. Skip to leave previous file or pass `None` + to clean it. + :param markup_auto_adjust: (BotX default: False) Move button to next + row, if its text doesn't fit. + """ + + method = EditEventMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIEditEventRequestPayload.from_domain( + sync_id=sync_id, + body=body, + metadata=metadata, + bubbles=bubbles, + keyboard=keyboard, + file=file, + markup_auto_adjust=markup_auto_adjust, + ) + + await method.execute(payload) + + async def reply( + self, + *, + message: ReplyMessage, + ) -> None: + """Reply message. + + :param message: Built outgoing reply message. + """ + + await self.reply_message( + bot_id=message.bot_id, + sync_id=message.sync_id, + body=message.body, + metadata=message.metadata, + bubbles=message.bubbles, + keyboard=message.keyboard, + file=message.file, + silent_response=message.silent_response, + markup_auto_adjust=message.markup_auto_adjust, + stealth_mode=message.stealth_mode, + send_push=message.send_push, + ignore_mute=message.ignore_mute, + ) + + async def reply_message( + self, + *, + bot_id: UUID, + sync_id: UUID, + body: str, + metadata: Missing[Dict[str, Any]] = Undefined, + bubbles: Missing[BubbleMarkup] = Undefined, + keyboard: Missing[KeyboardMarkup] = Undefined, + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined, + silent_response: Missing[bool] = Undefined, + markup_auto_adjust: Missing[bool] = Undefined, + stealth_mode: Missing[bool] = Undefined, + send_push: Missing[bool] = Undefined, + ignore_mute: Missing[bool] = Undefined, + ) -> None: + """Reply on message by `sync_id`. + + :param bot_id: Bot which should perform the request. + :param sync_id: `sync_id` of message to reply on. + :param body: Reply body. + :param metadata: Notification options. + :param bubbles: Bubbles (buttons attached to message) markup. + :param keyboard: Keyboard (buttons below message input) markup. + :param file: Attachment. + :param silent_response: (BotX default: False) Exclude next user + messages from history. + :param markup_auto_adjust: (BotX default: False) Move button to next + row, if its text doesn't fit. + :param stealth_mode: (BotX default: False) Enable stealth mode. + :param send_push: (BotX default: True) Send push notification on + devices. + :param ignore_mute: (BotX default: False) Ignore mute or dnd (do not + disturb). + """ + + payload = BotXAPIReplyEventRequestPayload.from_domain( + sync_id=sync_id, + body=body, + metadata=metadata, + bubbles=bubbles, + keyboard=keyboard, + file=file, + silent_response=silent_response, + markup_auto_adjust=markup_auto_adjust, + stealth_mode=stealth_mode, + send_push=send_push, + ignore_mute=ignore_mute, + ) + method = ReplyEventMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + await method.execute(payload) + + async def get_message_status(self, *, bot_id: UUID, sync_id: UUID) -> MessageStatus: + """ + Get status of message by `sync_id`. + + :param bot_id: Bot which should perform the request. + :param sync_id: `sync_id` of message to get its status. + + :returns: Message status object. + """ + payload = BotXAPIMessageStatusRequestPayload.from_domain(sync_id=sync_id) + method = MessageStatusMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + botx_api_message_status = await method.execute(payload) + return botx_api_message_status.to_domain() + + async def start_typing( + self, + *, + bot_id: UUID, + chat_id: UUID, + ) -> None: + """Send `typing` event. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + """ + + payload = BotXAPITypingEventRequestPayload.from_domain( + chat_id=chat_id, + ) + method = TypingEventMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + await method.execute(payload) + + async def stop_typing( + self, + *, + bot_id: UUID, + chat_id: UUID, + ) -> None: + """Send `stop_typing` event. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + """ + + payload = BotXAPIStopTypingEventRequestPayload.from_domain( + chat_id=chat_id, + ) + method = StopTypingEventMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + await method.execute(payload) + + # - Chats API - + async def list_chats( + self, + *, + bot_id: UUID, + ) -> List[ChatListItem]: + """Get all bot chats. + + :param bot_id: Bot which should perform the request. + + :returns: List of chats info. + """ + + method = ListChatsMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + botx_api_list_chat = await method.execute() + + return botx_api_list_chat.to_domain() + + async def chat_info( + self, + *, + bot_id: UUID, + chat_id: UUID, + ) -> ChatInfo: + """Get chat information. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + + :return: Chat information. + """ + + method = ChatInfoMethod(bot_id, self._httpx_client, self._bot_accounts_storage) + + payload = BotXAPIChatInfoRequestPayload.from_domain(chat_id=chat_id) + botx_api_chat_info = await method.execute(payload) + + return botx_api_chat_info.to_domain() + + async def add_users_to_chat( + self, + *, + bot_id: UUID, + chat_id: UUID, + huids: List[UUID], + ) -> None: + """Add user to chat. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param huids: List of eXpress account ids. + """ + + method = AddUserMethod(bot_id, self._httpx_client, self._bot_accounts_storage) + + payload = BotXAPIAddUserRequestPayload.from_domain(chat_id=chat_id, huids=huids) + await method.execute(payload) + + async def remove_users_from_chat( + self, + *, + bot_id: UUID, + chat_id: UUID, + huids: List[UUID], + ) -> None: + """Remove eXpress accounts from chat. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param huids: List of eXpress account ids. + """ + + method = RemoveUserMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + payload = BotXAPIRemoveUserRequestPayload.from_domain( + chat_id=chat_id, + huids=huids, + ) + await method.execute(payload) + + async def promote_to_chat_admins( + self, + *, + bot_id: UUID, + chat_id: UUID, + huids: List[UUID], + ) -> None: + """Promote users in chat to admins. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param huids: List of eXpress account ids. + """ + + method = AddAdminMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + payload = BotXAPIAddAdminRequestPayload.from_domain( + chat_id=chat_id, + huids=huids, + ) + await method.execute(payload) + + async def enable_stealth( + self, + *, + bot_id: UUID, + chat_id: UUID, + disable_web_client: Missing[bool] = Undefined, + ttl_after_read: Missing[int] = Undefined, + total_ttl: Missing[int] = Undefined, + ) -> None: + """Enable stealth mode. + + After the expiration of the time all messages will be hidden. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param disable_web_client: (BotX default: False) Should messages + be shown in web. + :param ttl_after_read: (BotX default: OFF) Time of messages burning + after read. + :param total_ttl: (BotX default: OFF) Time of messages burning after + send. + """ + + method = SetStealthMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISetStealthRequestPayload.from_domain( + chat_id=chat_id, + disable_web_client=disable_web_client, + ttl_after_read=ttl_after_read, + total_ttl=total_ttl, + ) + + await method.execute(payload) + + async def disable_stealth( + self, + *, + bot_id: UUID, + chat_id: UUID, + ) -> None: + """Disable stealth model. Hides all messages that were in stealth. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + """ + + method = DisableStealthMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIDisableStealthRequestPayload.from_domain(chat_id=chat_id) + + await method.execute(payload) + + async def create_chat( + self, + *, + bot_id: UUID, + name: str, + chat_type: ChatTypes, + huids: List[UUID], + description: Optional[str] = None, + shared_history: Missing[bool] = Undefined, + ) -> UUID: + """Create chat. + + :param bot_id: Bot which should perform the request. + :param name: Chat visible name. + :param chat_type: Chat type. + :param huids: List of eXpress account ids. + :param description: Chat description. + :param shared_history: (BotX default: False) Open old chat history for + new added users. + + :return: Created chat uuid. + """ + + method = CreateChatMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + payload = BotXAPICreateChatRequestPayload.from_domain( + name=name, + chat_type=chat_type, + huids=huids, + shared_history=shared_history, + description=description, + ) + botx_api_chat_id = await method.execute(payload) + + return botx_api_chat_id.to_domain() + + async def pin_message( + self, + *, + bot_id: UUID, + chat_id: UUID, + sync_id: UUID, + ) -> None: + """Pin message in chat. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param sync_id: Target sync id. + """ + + method = PinMessageMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIPinMessageRequestPayload.from_domain( + chat_id=chat_id, + sync_id=sync_id, + ) + + await method.execute(payload) + + async def unpin_message( + self, + *, + bot_id: UUID, + chat_id: UUID, + ) -> None: + """Unpin message in chat. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + """ + + method = UnpinMessageMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIUnpinMessageRequestPayload.from_domain(chat_id=chat_id) + + await method.execute(payload) + + # - Users API - + async def search_user_by_email( + self, + *, + bot_id: UUID, + email: str, + ) -> UserFromSearch: + """Search user by email for search. + + :param bot_id: Bot which should perform the request. + :param email: User email. + + :return: User information. + """ + + method = SearchUserByEmailMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISearchUserByEmailRequestPayload.from_domain(email=email) + + botx_api_user_from_search = await method.execute(payload) + + return botx_api_user_from_search.to_domain() + + async def search_user_by_huid( + self, + *, + bot_id: UUID, + huid: UUID, + ) -> UserFromSearch: + """Search user by huid for search. + + :param bot_id: Bot which should perform the request. + :param huid: User huid. + + :return: User information. + """ + + method = SearchUserByHUIDMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISearchUserByHUIDRequestPayload.from_domain(huid=huid) + + botx_api_user_from_search = await method.execute(payload) + + return botx_api_user_from_search.to_domain() + + async def search_user_by_ad( + self, + *, + bot_id: UUID, + ad_login: str, + ad_domain: str, + ) -> UserFromSearch: + """Search user by AD login and AD domain for search. + + :param bot_id: Bot which should perform the request. + :param ad_login: User AD login. + :param ad_domain: User AD domain. + + :return: User information. + """ + + method = SearchUserByLoginMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISearchUserByLoginRequestPayload.from_domain( + ad_login=ad_login, + ad_domain=ad_domain, + ) + + botx_api_user_from_search = await method.execute(payload) + + return botx_api_user_from_search.to_domain() + + # - SmartApps API - + async def send_smartapp_event( + self, + *, + bot_id: UUID, + chat_id: UUID, + data: Dict[str, Any], + ref: MissingOptional[UUID] = Undefined, + opts: Missing[Dict[str, Any]] = Undefined, + files: Missing[List[File]] = Undefined, + ) -> None: + """Send SmartApp event. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param data: Event payload. + :param ref: Request identifier. + :param opts: Event options. + :param files: Files. + """ + + method = SmartAppEventMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISmartAppEventRequestPayload.from_domain( + ref=ref, + smartapp_id=bot_id, + chat_id=chat_id, + data=data, + opts=opts, + files=files, + ) + + await method.execute(payload) + + async def send_smartapp_notification( + self, + bot_id: UUID, + chat_id: UUID, + smartapp_counter: int, + opts: Missing[Dict[str, Any]] = Undefined, + ) -> None: + """Send SmartApp notification. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param smartapp_counter: Value app's counter. + :param opts: Vvent options. + """ + + method = SmartAppNotificationMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPISmartAppNotificationRequestPayload.from_domain( + chat_id=chat_id, + smartapp_counter=smartapp_counter, + opts=opts, + ) + + await method.execute(payload) + + # - Stickers API - + async def create_sticker_pack(self, *, bot_id: UUID, name: str) -> StickerPack: + """Create empty sticker pack. + + :param bot_id: Bot which should perform the request. + :param name: Sticker pack name. + + :return: Created sticker pack. + """ + + method = CreateStickerPackMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPICreateStickerPackRequestPayload.from_domain(name=name) + + botx_api_sticker_pack = await method.execute(payload) + + return botx_api_sticker_pack.to_domain() + + async def add_sticker( + self, + *, + bot_id: UUID, + sticker_pack_id: UUID, + emoji: str, + async_buffer: AsyncBufferReadable, + ) -> Sticker: + """Add sticker in sticker pack. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Sticker pack id to indicate where to add. + :param emoji: Sticker emoji. + :param async_buffer: Sticker image file. Only PNG. + + :return: Added sticker. + """ + + await ensure_file_content_is_png(async_buffer) + await ensure_sticker_image_size_valid(async_buffer) + + method = AddStickerMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = await BotXAPIAddStickerRequestPayload.from_domain( + sticker_pack_id=sticker_pack_id, + emoji=emoji, + async_buffer=async_buffer, + ) + + botx_api_sticker = await method.execute(payload) + + return botx_api_sticker.to_domain() + + async def delete_sticker( + self, + *, + bot_id: UUID, + sticker_pack_id: UUID, + sticker_id: UUID, + ) -> None: + """Delete sticker from sticker pack. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Target sticker pack id. + :param sticker_id: Sticker id which should be deleted. + """ + + method = DeleteStickerMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = await BotXAPIDeleteStickerRequestPayload.from_domain( + sticker_id=sticker_id, + sticker_pack_id=sticker_pack_id, + ) + + await method.execute(payload) + + async def iterate_by_sticker_packs( + self, + *, + bot_id: UUID, + user_huid: UUID, + ) -> AsyncIterable[StickerPackFromList]: + """Iterate by user sticker packs. + + :param bot_id: Bot which should perform the request. + :param user_huid: User huid. + + :yield: Sticker pack. + """ + + after = None + + method = GetStickerPacksMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + while True: + payload = BotXAPIGetStickerPacksRequestPayload.from_domain( + huid=user_huid, + limit=STICKER_PACKS_PER_PAGE, + after=after, + ) + botx_api_sticker_pack_list = await method.execute(payload) + + sticker_pack_page = botx_api_sticker_pack_list.to_domain() + after = sticker_pack_page.after + + for sticker_pack in sticker_pack_page.sticker_packs: + yield sticker_pack + + if not after: + break + + async def get_sticker_pack( + self, + *, + bot_id: UUID, + sticker_pack_id: UUID, + ) -> StickerPack: + """Get sticker pack. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Sticker pack id. + + :return: Sticker pack. + """ + + method = GetStickerPackMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIGetStickerPackRequestPayload.from_domain( + sticker_pack_id=sticker_pack_id, + ) + + botx_api_sticker_pack = await method.execute(payload) + + return botx_api_sticker_pack.to_domain() + + async def delete_sticker_pack(self, *, bot_id: UUID, sticker_pack_id: UUID) -> None: + """Delete existing sticker pack. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Target sticker pack. + """ + + method = DeleteStickerPackMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + + payload = BotXAPIDeleteStickerPackRequestPayload.from_domain( + sticker_pack_id=sticker_pack_id, + ) + + await method.execute(payload) + + async def get_sticker( + self, + *, + bot_id: UUID, + sticker_pack_id: UUID, + sticker_id: UUID, + ) -> Sticker: + """Get sticker. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Sticker pack id. + :param sticker_id: Sticker id. + + :return: Sticker. + """ + + method = GetStickerMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIGetStickerRequestPayload.from_domain( + sticker_pack_id=sticker_pack_id, + sticker_id=sticker_id, + ) + + botx_api_sticker = await method.execute(payload) + + return botx_api_sticker.to_domain() + + async def edit_sticker_pack( + self, + *, + bot_id: UUID, + sticker_pack_id: UUID, + name: str, + preview: UUID, + stickers_order: List[UUID], + ) -> StickerPack: + """Edit Sticker pack. + + :param bot_id: Bot which should perform the request. + :param sticker_pack_id: Sticker pack id. + :param name: Sticker pack name. + :param preview: Sticker from the set selected as a preview. + :param stickers_order: Sticker IDs in order they are displayed. + + :return: Edited sticker pack. + """ + + method = EditStickerPackMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIEditStickerPackRequestPayload.from_domain( + sticker_pack_id=sticker_pack_id, + name=name, + preview=preview, + stickers_order=stickers_order, + ) + + botx_api_sticker_pack = await method.execute(payload) + + return botx_api_sticker_pack.to_domain() + + # - Files API - + async def download_file( + self, + *, + bot_id: UUID, + chat_id: UUID, + file_id: UUID, + async_buffer: AsyncBufferWritable, + ) -> None: + """Download file form file service. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param file_id: Async file id. + :param async_buffer: Buffer to write downloaded file. + """ + + method = DownloadFileMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIDownloadFileRequestPayload.from_domain( + chat_id=chat_id, + file_id=file_id, + ) + + await method.execute(payload, async_buffer) + + async def upload_file( + self, + *, + bot_id: UUID, + chat_id: UUID, + async_buffer: AsyncBufferReadable, + filename: str, + duration: Missing[int] = Undefined, + caption: Missing[str] = Undefined, + ) -> File: + """Upload file to file service. + + :param bot_id: Bot which should perform the request. + :param chat_id: Target chat id. + :param async_buffer: Buffer to write downloaded file. + :param filename: File name. + :param duration: Video duration. + :param caption: Text under file. + + :return: Meta info of uploaded file. + """ + + method = UploadFileMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + payload = BotXAPIUploadFileRequestPayload.from_domain( + chat_id=chat_id, + duration=duration, + caption=caption, + ) + + botx_api_async_file = await method.execute(payload, async_buffer, filename) + + return botx_api_async_file.to_domain() + + @staticmethod + def _build_main_collector( + collectors: Sequence[HandlerCollector], + middlewares: List[Middleware], + exception_handlers: Optional[ExceptionHandlersDict] = None, + ) -> HandlerCollector: + main_collector = HandlerCollector(middlewares=middlewares) + main_collector.insert_exception_middleware(exception_handlers) + main_collector.include(*collectors) + + return main_collector diff --git a/botx/bot/bot_accounts_storage.py b/botx/bot/bot_accounts_storage.py new file mode 100644 index 00000000..d97cbcdd --- /dev/null +++ b/botx/bot/bot_accounts_storage.py @@ -0,0 +1,48 @@ +import base64 +import hashlib +import hmac +from typing import Dict, Iterator, List, Optional +from uuid import UUID + +from botx.bot.exceptions import UnknownBotAccountError +from botx.models.bot_account import BotAccount, BotAccountWithSecret + + +class BotAccountsStorage: + def __init__(self, bot_accounts: List[BotAccountWithSecret]) -> None: + self._bot_accounts = bot_accounts + self._auth_tokens: Dict[UUID, str] = {} + + def iter_bot_accounts(self) -> Iterator[BotAccount]: + yield from self._bot_accounts + + def get_host(self, bot_id: UUID) -> str: + bot_account = self._get_bot_account(bot_id) + return bot_account.host + + def set_token(self, bot_id: UUID, token: str) -> None: + self._auth_tokens[bot_id] = token + + def get_token_or_none(self, bot_id: UUID) -> Optional[str]: + return self._auth_tokens.get(bot_id) + + def build_signature(self, bot_id: UUID) -> str: + bot_account = self._get_bot_account(bot_id) + + signed_bot_id = hmac.new( + key=bot_account.secret_key.encode(), + msg=str(bot_account.id).encode(), + digestmod=hashlib.sha256, + ).digest() + + return base64.b16encode(signed_bot_id).decode() + + def ensure_bot_id_exists(self, bot_id: UUID) -> None: + self._get_bot_account(bot_id) + + def _get_bot_account(self, bot_id: UUID) -> BotAccountWithSecret: + for bot_account in self._bot_accounts: + if bot_account.id == bot_id: + return bot_account + + raise UnknownBotAccountError(bot_id) diff --git a/botx/bot/callbacks_manager.py b/botx/bot/callbacks_manager.py new file mode 100644 index 00000000..c4a3f9fc --- /dev/null +++ b/botx/bot/callbacks_manager.py @@ -0,0 +1,63 @@ +import asyncio +from typing import TYPE_CHECKING, Dict, Optional +from uuid import UUID + +from botx.bot.exceptions import BotShuttingDownError, BotXMethodCallbackNotFoundError +from botx.client.exceptions.callbacks import CallbackNotReceivedError +from botx.logger import logger +from botx.models.method_callbacks import BotXMethodCallback + +if TYPE_CHECKING: + from asyncio import Future # noqa: WPS458 + + +class CallbacksManager: + def __init__(self) -> None: + self._callback_futures: Dict[UUID, "Future[BotXMethodCallback]"] = {} + + def create_botx_method_callback(self, sync_id: UUID) -> None: + self._callback_futures[sync_id] = asyncio.Future() + + def set_botx_method_callback_result( + self, + callback: BotXMethodCallback, + ) -> None: + sync_id = callback.sync_id + future = self._pop_future(sync_id) + + if future.cancelled(): + logger.warning( + f"BotX method with sync_id `{sync_id!s}` don't wait callback", + ) + return + + future.set_result(callback) + + async def wait_botx_method_callback( + self, + sync_id: UUID, + timeout: Optional[int], + ) -> BotXMethodCallback: + future = self._callback_futures[sync_id] + + try: + return await asyncio.wait_for(future, timeout=timeout) + except asyncio.TimeoutError as exc: + raise CallbackNotReceivedError(sync_id) from exc + + def stop_callbacks_waiting(self) -> None: + for sync_id, future in self._callback_futures.items(): + if not future.done(): + future.set_exception( + BotShuttingDownError( + f"Callback with sync_id `{sync_id!s}` can't be received", + ), + ) + + def _pop_future(self, sync_id: UUID) -> "Future[BotXMethodCallback]": + try: + future = self._callback_futures.pop(sync_id) + except KeyError: + raise BotXMethodCallbackNotFoundError(sync_id) from None + + return future diff --git a/botx/bot/contextvars.py b/botx/bot/contextvars.py new file mode 100644 index 00000000..030a50c7 --- /dev/null +++ b/botx/bot/contextvars.py @@ -0,0 +1,10 @@ +from contextvars import ContextVar +from typing import TYPE_CHECKING +from uuid import UUID + +if TYPE_CHECKING: # To avoid circular import + from botx.bot.bot import Bot + +bot_var: ContextVar["Bot"] = ContextVar("bot_var") +bot_id_var: ContextVar[UUID] = ContextVar("bot_id") +chat_id_var: ContextVar[UUID] = ContextVar("chat_id") diff --git a/botx/bot/exceptions.py b/botx/bot/exceptions.py new file mode 100644 index 00000000..b6313189 --- /dev/null +++ b/botx/bot/exceptions.py @@ -0,0 +1,29 @@ +from typing import Any +from uuid import UUID + + +class UnknownBotAccountError(Exception): + def __init__(self, bot_id: UUID) -> None: + self.bot_id = bot_id + self.message = f"No bot account with bot_id: `{bot_id!s}`" + super().__init__(self.message) + + +class BotXMethodCallbackNotFoundError(Exception): + def __init__(self, sync_id: UUID) -> None: + self.sync_id = sync_id + self.message = f"No callback found with sync_id: `{sync_id!s}`" + super().__init__(self.message) + + +class BotShuttingDownError(Exception): + def __init__(self, context: Any) -> None: + self.context = context + self.message = f"Bot is shutting down: {context}" + super().__init__(self.message) + + +class AnswerDestinationLookupError(Exception): + def __init__(self) -> None: + self.message = "No IncomingMessage received. Use `Bot.send` instead" + super().__init__(self.message) diff --git a/botx/bot/handler.py b/botx/bot/handler.py new file mode 100644 index 00000000..b5ac9ffc --- /dev/null +++ b/botx/bot/handler.py @@ -0,0 +1,79 @@ +from dataclasses import dataclass +from functools import partial +from typing import TYPE_CHECKING, Awaitable, Callable, List, Literal, TypeVar, Union + +from botx.models.commands import BotCommand +from botx.models.message.incoming_message import IncomingMessage +from botx.models.status import StatusRecipient +from botx.models.system_events.added_to_chat import AddedToChatEvent +from botx.models.system_events.chat_created import ChatCreatedEvent +from botx.models.system_events.cts_login import CTSLoginEvent +from botx.models.system_events.cts_logout import CTSLogoutEvent +from botx.models.system_events.deleted_from_chat import DeletedFromChatEvent +from botx.models.system_events.internal_bot_notification import ( + InternalBotNotificationEvent, +) +from botx.models.system_events.left_from_chat import LeftFromChatEvent +from botx.models.system_events.smartapp_event import SmartAppEvent + +if TYPE_CHECKING: # To avoid circular import + from botx.bot.bot import Bot + +TBotCommand = TypeVar("TBotCommand", bound=BotCommand) +HandlerFunc = Callable[[TBotCommand, "Bot"], Awaitable[None]] + +IncomingMessageHandlerFunc = HandlerFunc[IncomingMessage] +SystemEventHandlerFunc = Union[ + HandlerFunc[AddedToChatEvent], + HandlerFunc[ChatCreatedEvent], + HandlerFunc[DeletedFromChatEvent], + HandlerFunc[LeftFromChatEvent], + HandlerFunc[CTSLoginEvent], + HandlerFunc[CTSLogoutEvent], + HandlerFunc[InternalBotNotificationEvent], + HandlerFunc[SmartAppEvent], +] + +VisibleFunc = Callable[[StatusRecipient, "Bot"], Awaitable[bool]] + +Middleware = Callable[ + [IncomingMessage, "Bot", IncomingMessageHandlerFunc], + Awaitable[None], +] + + +@dataclass +class BaseIncomingMessageHandler: + handler_func: IncomingMessageHandlerFunc + middlewares: List[Middleware] + + async def __call__(self, message: IncomingMessage, bot: "Bot") -> None: + handler_func = self.handler_func + + for middleware in self.middlewares[::-1]: + handler_func = partial(middleware, call_next=handler_func) + + await handler_func(message, bot) + + def add_middlewares(self, middlewares: List[Middleware]) -> None: + self.middlewares = middlewares + self.middlewares + + +@dataclass +class HiddenCommandHandler(BaseIncomingMessageHandler): + # Default should be here, see: https://github.com/python/mypy/issues/6113 + visible: Literal[False] = False + + +@dataclass +class VisibleCommandHandler(BaseIncomingMessageHandler): + description: str + visible: Union[Literal[True], VisibleFunc] = True + + +@dataclass +class DefaultMessageHandler(BaseIncomingMessageHandler): + """Just for separate type.""" + + +CommandHandler = Union[HiddenCommandHandler, VisibleCommandHandler] diff --git a/botx/bot/handler_collector.py b/botx/bot/handler_collector.py new file mode 100644 index 00000000..438c4fb3 --- /dev/null +++ b/botx/bot/handler_collector.py @@ -0,0 +1,438 @@ +import asyncio +import re +from typing import ( + TYPE_CHECKING, + Callable, + Dict, + List, + Optional, + Sequence, + Type, + Union, + overload, +) +from weakref import WeakSet + +from botx.bot.contextvars import bot_id_var, bot_var, chat_id_var +from botx.bot.handler import ( + CommandHandler, + DefaultMessageHandler, + HandlerFunc, + HiddenCommandHandler, + IncomingMessageHandlerFunc, + Middleware, + SystemEventHandlerFunc, + VisibleCommandHandler, + VisibleFunc, +) +from botx.bot.middlewares.exception_middleware import ( + ExceptionHandlersDict, + ExceptionMiddleware, +) +from botx.converters import optional_sequence_to_list +from botx.logger import logger +from botx.models.commands import BotCommand, SystemEvent +from botx.models.message.incoming_message import IncomingMessage +from botx.models.status import BotMenu, StatusRecipient +from botx.models.system_events.added_to_chat import AddedToChatEvent +from botx.models.system_events.chat_created import ChatCreatedEvent +from botx.models.system_events.cts_login import CTSLoginEvent +from botx.models.system_events.cts_logout import CTSLogoutEvent +from botx.models.system_events.deleted_from_chat import DeletedFromChatEvent +from botx.models.system_events.internal_bot_notification import ( + InternalBotNotificationEvent, +) +from botx.models.system_events.left_from_chat import LeftFromChatEvent +from botx.models.system_events.smartapp_event import SmartAppEvent + +if TYPE_CHECKING: # To avoid circular import + from botx.bot.bot import Bot + + +class HandlerCollector: + VALID_COMMAND_NAME_RE = re.compile(r"^\/[^\s\/]+$", flags=re.UNICODE) + + def __init__(self, middlewares: Optional[Sequence[Middleware]] = None) -> None: + self._user_commands_handlers: Dict[str, CommandHandler] = {} + self._default_message_handler: Optional[DefaultMessageHandler] = None + self._system_events_handlers: Dict[ + Type[BotCommand], + SystemEventHandlerFunc, + ] = {} + self._middlewares = optional_sequence_to_list(middlewares) + self._tasks: "WeakSet[asyncio.Task[None]]" = WeakSet() + + def include(self, *others: "HandlerCollector") -> None: + """Include other `HandlerCollector`.""" + + for collector in others: + self._include_collector(collector) + + def async_handle_bot_command( + self, + bot: "Bot", + bot_command: BotCommand, + ) -> None: + task = asyncio.create_task( + self.handle_bot_command(bot_command, bot), + ) + self._tasks.add(task) + + async def handle_incoming_message_by_command( + self, + message: IncomingMessage, + bot: "Bot", + command: str, + ) -> None: + message_handler = self._get_command_handler(command) + if message_handler: + self._fill_contextvars(message, bot) + await message_handler(message, bot) + + async def handle_bot_command(self, bot_command: BotCommand, bot: "Bot") -> None: + if isinstance(bot_command, IncomingMessage): + message_handler = self._get_incoming_message_handler(bot_command) + if message_handler: + self._fill_contextvars(bot_command, bot) + await message_handler(bot_command, bot) + + elif isinstance( + bot_command, + # TODO: Replace `__args__` with `typing.get_origin` on python 3.7 drop. + SystemEvent.__args__, # type: ignore [attr-defined] # noqa: WPS609 + ): + event_handler = self._get_system_event_handler_or_none(bot_command) + if event_handler: + self._fill_contextvars(bot_command, bot) + await event_handler(bot_command, bot) + + else: + raise NotImplementedError(f"Unsupported event type: `{bot_command}`") + + async def get_bot_menu( + self, + status_recipient: StatusRecipient, + bot: "Bot", + ) -> BotMenu: + bot_menu = {} + + for command_name, handler in self._user_commands_handlers.items(): + if handler.visible is True or ( + callable(handler.visible) + and await handler.visible(status_recipient, bot) + ): + bot_menu[command_name] = handler.description + + return BotMenu(bot_menu) + + def command( + self, + command_name: str, + visible: Union[bool, VisibleFunc] = True, + description: Optional[str] = None, + middlewares: Optional[Sequence[Middleware]] = None, + ) -> Callable[[IncomingMessageHandlerFunc], IncomingMessageHandlerFunc]: + """Decorate command handler.""" + + if not self.VALID_COMMAND_NAME_RE.match(command_name): + raise ValueError("Command should start with '/' and doesn't include spaces") + + def decorator( + handler_func: IncomingMessageHandlerFunc, + ) -> IncomingMessageHandlerFunc: + if command_name in self._user_commands_handlers: + raise ValueError( + f"Handler for command `{command_name}` already registered", + ) + + self._user_commands_handlers[command_name] = self._build_command_handler( + handler_func, + visible, + description, + self._middlewares + optional_sequence_to_list(middlewares), + ) + + return handler_func + + return decorator + + @overload + def default_message_handler( + self, + handler_func: IncomingMessageHandlerFunc, + ) -> IncomingMessageHandlerFunc: + ... # noqa: WPS428 + + @overload + def default_message_handler( + self, + *, + middlewares: Optional[Sequence[Middleware]] = None, + ) -> Callable[[IncomingMessageHandlerFunc], IncomingMessageHandlerFunc]: + ... # noqa: WPS428 + + def default_message_handler( # noqa: WPS320 + self, + handler_func: Optional[IncomingMessageHandlerFunc] = None, + *, + middlewares: Optional[Sequence[Middleware]] = None, + ) -> Union[ + IncomingMessageHandlerFunc, + Callable[[IncomingMessageHandlerFunc], IncomingMessageHandlerFunc], + ]: + """Decorate fallback messages handler.""" + if self._default_message_handler: + raise ValueError("Default command handler already registered") + + def decorator( + handler_func: IncomingMessageHandlerFunc, # noqa: WPS442 + ) -> IncomingMessageHandlerFunc: + self._default_message_handler = DefaultMessageHandler( + handler_func=handler_func, + middlewares=self._middlewares + optional_sequence_to_list(middlewares), + ) + + return handler_func + + if callable(handler_func) and not middlewares: + return decorator(handler_func) + + return decorator + + def chat_created( + self, + handler_func: HandlerFunc[ChatCreatedEvent], + ) -> HandlerFunc[ChatCreatedEvent]: + """Decorate `chat_created` event handler.""" + + self._system_event(ChatCreatedEvent, handler_func) + + return handler_func + + def added_to_chat( + self, + handler_func: HandlerFunc[AddedToChatEvent], + ) -> HandlerFunc[AddedToChatEvent]: + """Decorate `added_to_chat` event handler.""" + + self._system_event(AddedToChatEvent, handler_func) + + return handler_func + + def deleted_from_chat( + self, + handler_func: HandlerFunc[DeletedFromChatEvent], + ) -> HandlerFunc[DeletedFromChatEvent]: + """Decorate `deleted_from_chat` event handler.""" + + self._system_event(DeletedFromChatEvent, handler_func) + + return handler_func + + def left_from_chat( + self, + handler_func: HandlerFunc[LeftFromChatEvent], + ) -> HandlerFunc[LeftFromChatEvent]: + """Decorate `left_from_chat` event handler.""" + + self._system_event(LeftFromChatEvent, handler_func) + + return handler_func + + def internal_bot_notification( + self, + handler_func: HandlerFunc[InternalBotNotificationEvent], + ) -> HandlerFunc[InternalBotNotificationEvent]: + """Decorate `internal_bot_notification` event handler.""" + + self._system_event(InternalBotNotificationEvent, handler_func) + + return handler_func + + def cts_login( + self, + handler_func: HandlerFunc[CTSLoginEvent], + ) -> HandlerFunc[CTSLoginEvent]: + """Decorate `cts_login` event handler.""" + + self._system_event(CTSLoginEvent, handler_func) + + return handler_func + + def cts_logout( + self, + handler_func: HandlerFunc[CTSLogoutEvent], + ) -> HandlerFunc[CTSLogoutEvent]: + """Decorate `cts_logout` event handler.""" + + self._system_event(CTSLogoutEvent, handler_func) + + return handler_func + + def smartapp_event( + self, + handler_func: HandlerFunc[SmartAppEvent], + ) -> HandlerFunc[SmartAppEvent]: + """Decorate `smartapp` event handler.""" + + self._system_event(SmartAppEvent, handler_func) + + return handler_func + + def insert_exception_middleware( + self, + exception_handlers: Optional[ExceptionHandlersDict] = None, + ) -> None: + exception_middleware = ExceptionMiddleware(exception_handlers or {}) + self._middlewares.insert(0, exception_middleware.dispatch) + + async def wait_active_tasks(self) -> None: + if self._tasks: + await asyncio.wait( + self._tasks, + return_when=asyncio.ALL_COMPLETED, + ) + + def _include_collector(self, other: "HandlerCollector") -> None: + # - Message handlers - + command_duplicates = set(self._user_commands_handlers) & set( + other._user_commands_handlers, + ) + if command_duplicates: + raise ValueError( + f"Handlers for {command_duplicates} commands already registered", + ) + + other_handlers = other._user_commands_handlers + for handler in other_handlers.values(): + handler.add_middlewares(self._middlewares) + + self._user_commands_handlers.update(other_handlers) + + # - Default message handler - + if self._default_message_handler and other._default_message_handler: + raise ValueError("Default message handler already registered") + + if not self._default_message_handler and other._default_message_handler: + other._default_message_handler.add_middlewares(self._middlewares) + self._default_message_handler = other._default_message_handler + + # - System events - + events_duplicates = set(self._system_events_handlers) & set( + other._system_events_handlers, + ) + if events_duplicates: + raise ValueError( + f"Handlers for {events_duplicates} events already registered", + ) + + self._system_events_handlers.update(other._system_events_handlers) + + def _get_incoming_message_handler( + self, + message: IncomingMessage, + ) -> Union[CommandHandler, DefaultMessageHandler, None]: + return self._get_command_handler(message.body) + + def _get_command_handler( + self, + command: str, + ) -> Union[CommandHandler, DefaultMessageHandler, None]: + handler: Optional[Union[CommandHandler, DefaultMessageHandler]] = None + + command_name = self._get_command_name(command) + if command_name: + handler = self._user_commands_handlers.get(command_name) + if handler: + logger.info(f"Found handler for command `{command_name}`") + return handler + + if self._default_message_handler: + self._log_default_handler_call(command_name) + return self._default_message_handler + + logger.warning(f"Handler for message text `{command}` not found") + return None + + def _get_system_event_handler_or_none( + self, + event: SystemEvent, + ) -> Optional[SystemEventHandlerFunc]: + event_cls = event.__class__ + + handler = self._system_events_handlers.get(event_cls) + self._log_system_event_handler_call(event_cls.__name__, handler) + + return handler + + def _get_command_name(self, body: str) -> Optional[str]: + if not body: + return None + + command_name = body.split(maxsplit=1)[0] + if self.VALID_COMMAND_NAME_RE.match(command_name): + return command_name + + return None + + def _build_command_handler( + self, + handler_func: IncomingMessageHandlerFunc, + visible: Union[bool, VisibleFunc], + description: Optional[str], + middlewares: List[Middleware], + ) -> CommandHandler: + if visible is True or callable(visible): + if not description: + raise ValueError("Description is required for visible command") + + return VisibleCommandHandler( + handler_func=handler_func, + visible=visible, + description=description, + middlewares=middlewares, + ) + + return HiddenCommandHandler( + handler_func=handler_func, + middlewares=middlewares, + ) + + def _system_event( + self, + event_cls_name: Type[BotCommand], + handler_func: SystemEventHandlerFunc, + ) -> SystemEventHandlerFunc: + if event_cls_name in self._system_events_handlers: + raise ValueError(f"Handler for {event_cls_name} already registered") + + self._system_events_handlers[event_cls_name] = handler_func + + return handler_func + + def _fill_contextvars(self, bot_command: BotCommand, bot: "Bot") -> None: + bot_var.set(bot) + bot_id_var.set(bot_command.bot.id) + + chat = getattr(bot_command, "chat", None) + if chat: + chat_id_var.set(chat.id) + + def _log_system_event_handler_call( + self, + event_cls_name: str, + handler: Optional[SystemEventHandlerFunc], + ) -> None: + if handler: + logger.info(f"Found handler for `{event_cls_name}`") + else: + logger.info(f"Handler for `{event_cls_name}` not found") + + def _log_default_handler_call(self, command_name: Optional[str]) -> None: + if command_name: + logger.info( + f"Handler for command `{command_name}` not found, " + "using default handler", + ) + else: + logger.info("No command found, using default handler") diff --git a/docs/src/development/collector/collector0/__init__.py b/botx/bot/middlewares/__init__.py similarity index 100% rename from docs/src/development/collector/collector0/__init__.py rename to botx/bot/middlewares/__init__.py diff --git a/botx/bot/middlewares/exception_middleware.py b/botx/bot/middlewares/exception_middleware.py new file mode 100644 index 00000000..ae1a0055 --- /dev/null +++ b/botx/bot/middlewares/exception_middleware.py @@ -0,0 +1,56 @@ +from typing import TYPE_CHECKING, Awaitable, Callable, Dict, Optional, Type + +from botx.bot.handler import IncomingMessageHandlerFunc +from botx.logger import logger +from botx.models.message.incoming_message import IncomingMessage + +if TYPE_CHECKING: # To avoid circular import + from botx.bot.bot import Bot + +ExceptionHandler = Callable[ + [IncomingMessage, "Bot", Exception], + Awaitable[None], +] +ExceptionHandlersDict = Dict[Type[Exception], ExceptionHandler] + + +class ExceptionMiddleware: + """Exception handling middleware.""" + + def __init__(self, exception_handlers: ExceptionHandlersDict) -> None: + self._exception_handlers = exception_handlers + + async def dispatch( + self, + message: IncomingMessage, + bot: "Bot", + call_next: IncomingMessageHandlerFunc, + ) -> None: + try: + await call_next(message, bot) + except Exception as message_handler_exc: + exception_handler = self._get_exception_handler(message_handler_exc) + if exception_handler is None: + exc_name = type(message_handler_exc).__name__ + logger.exception( + f"Uncaught exception {exc_name}:", + message_handler_exc, + ) + return + + try: # noqa: WPS505 + await exception_handler(message, bot, message_handler_exc) + except Exception as error_handler_exc: + exc_name = type(message_handler_exc).__name__ + logger.exception( + f"Uncaught exception {exc_name} in exception handler:", + error_handler_exc, + ) + + def _get_exception_handler(self, exc: Exception) -> Optional[ExceptionHandler]: + for exc_cls in type(exc).mro(): + handler = self._exception_handlers.get(exc_cls) + if handler: + return handler + + return None diff --git a/botx/bot/testing.py b/botx/bot/testing.py new file mode 100644 index 00000000..59427a27 --- /dev/null +++ b/botx/bot/testing.py @@ -0,0 +1,14 @@ +from contextlib import asynccontextmanager +from typing import AsyncGenerator + +from botx.bot.bot import Bot + + +@asynccontextmanager +async def lifespan_wrapper(bot: Bot) -> AsyncGenerator[Bot, None]: + await bot.startup() + + try: + yield bot + finally: + await bot.shutdown() diff --git a/botx/bots/__init__.py b/botx/bots/__init__.py deleted file mode 100644 index 39810ec5..00000000 --- a/botx/bots/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for Bot and it's components.""" diff --git a/botx/bots/bots.py b/botx/bots/bots.py deleted file mode 100644 index 35c4595b..00000000 --- a/botx/bots/bots.py +++ /dev/null @@ -1,153 +0,0 @@ -"""Implementation for bot classes.""" - -import asyncio -from dataclasses import InitVar, field -from typing import Any, Callable, Dict, List -from weakref import WeakSet - -from loguru import logger -from pydantic.dataclasses import dataclass - -from botx import concurrency, exception_handlers, exceptions, shared, typing -from botx.bots.mixins import ( - clients, - collectors, - exceptions as exception_mixin, - lifespan, - middlewares, -) -from botx.clients.clients import async_client, sync_client as synchronous_client -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends -from botx.middlewares.authorization import AuthorizationMiddleware -from botx.middlewares.exceptions import ExceptionMiddleware -from botx.models import credentials, datastructures, menu -from botx.models.messages.message import Message - - -@dataclass(config=shared.BotXDataclassConfig) -class Bot( # noqa: WPS215 - collectors.BotCollectingMixin, - clients.ClientsMixin, - lifespan.LifespanMixin, - middlewares.MiddlewareMixin, - exception_mixin.ExceptionHandlersMixin, -): - """Class that implements bot behaviour.""" - - dependencies: InitVar[List[Depends]] = field(default=None) - bot_accounts: List[credentials.BotXCredentials] = field(default_factory=list) - startup_events: List[typing.BotLifespanEvent] = field(default_factory=list) - shutdown_events: List[typing.BotLifespanEvent] = field(default_factory=list) - - client: async_client.AsyncClient = field(init=False) - sync_client: synchronous_client.Client = field(init=False) - collector: Collector = field(init=False) - exception_middleware: ExceptionMiddleware = field(init=False) - state: datastructures.State = field(init=False) - dependency_overrides: Dict[Callable, Callable] = field( - init=False, - default_factory=dict, - ) - - tasks: WeakSet = field(init=False, default_factory=WeakSet) - - async def __call__(self, message: Message) -> None: - """Iterate through collector, find handler and execute it, running middlewares. - - Arguments: - message: message that will be proceed by handler. - """ - self.tasks.add(asyncio.ensure_future(self.exception_middleware(message))) - - def __post_init__(self, dependencies: List[Depends]) -> None: - """Initialize special fields. - - Arguments: - dependencies: initial background dependencies for inner collector. - """ - self.state = datastructures.State() - self.client = async_client.AsyncClient() - self.sync_client = synchronous_client.Client() - self.collector = Collector( - dependencies=dependencies, - dependency_overrides_provider=self, - ) - self.exception_middleware = ExceptionMiddleware(self.collector) - - self.add_exception_handler( - exceptions.DependencyFailure, - exception_handlers.dependency_failure_exception_handler, - ) - self.add_exception_handler( - exceptions.NoMatchFound, - exception_handlers.no_match_found_exception_handler, - ) - self.add_middleware(AuthorizationMiddleware) - - async def status(self, *args: Any, **kwargs: Any) -> menu.Status: - """Generate status object that could be return to BotX API on `/status`. - - Arguments: - args: additional positional arguments that will be passed to callable - status function. - kwargs: additional key arguments that will be passed to callable - status function. - - Returns: - Built status for returning to BotX API. - """ - status = menu.Status() - for handler in self.handlers: - if callable(handler.include_in_status): - include_in_status = await concurrency.callable_to_coroutine( - handler.include_in_status, - *args, - **kwargs, - ) - else: - include_in_status = handler.include_in_status - - if include_in_status: - status.result.commands.append( - menu.MenuCommand( - description=handler.description or "", - body=handler.body, - name=handler.name, - ), - ) - - return status - - async def execute_command(self, message: dict) -> None: - """Process data with incoming message and handle command inside. - - Arguments: - message: incoming message to bot. - """ - logger.bind(botx_bot=True, payload=message).debug("process incoming message") - msg = Message.from_dict(message, self) - - # raise UnknownBotError if not registered. - self.get_account_by_bot_id(msg.bot_id) - - await self(msg) - - async def authorize(self, *args: Any) -> None: - """Process auth for each bot account.""" - for account in self.bot_accounts: - try: - token = await self.get_token( - account.host, - account.bot_id, - account.signature, - ) - except (exceptions.BotXAPIError, exceptions.BotXConnectError) as exc: - logger.bind(botx_bot=True).warning( - f"Credentials `host - {account.host}, " # noqa: WPS305 - f"bot_id - {account.bot_id}` are invalid. " - f"Reason - {exc.message_template}", - ) - continue - - account.token = token diff --git a/botx/bots/mixins/__init__.py b/botx/bots/mixins/__init__.py deleted file mode 100644 index bffd8c59..00000000 --- a/botx/bots/mixins/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for mixins for bot.""" diff --git a/botx/bots/mixins/clients.py b/botx/bots/mixins/clients.py deleted file mode 100644 index 506694a1..00000000 --- a/botx/bots/mixins/clients.py +++ /dev/null @@ -1,50 +0,0 @@ -"""Define Mixin that combines API and sending mixins and will be used in Bot.""" -from typing import List -from uuid import UUID - -from botx.bots.mixins.requests.mixin import BotXRequestsMixin -from botx.bots.mixins.sending import SendingMixin -from botx.exceptions import TokenError, UnknownBotError -from botx.models.credentials import BotXCredentials - - -class ClientsMixin(SendingMixin, BotXRequestsMixin): - """Mixin that defines methods that are used for communicating with BotX API.""" - - bot_accounts: List[BotXCredentials] - - def get_account_by_bot_id(self, bot_id: UUID) -> BotXCredentials: - """Find BotCredentials in bot registered bot. - - Arguments: - bot_id: UUID of bot for which server should be searched. - - Returns: - Found instance of registered server. - - Raises: - UnknownBotError: raised if account was not found. - """ - for bot in self.bot_accounts: - if bot.bot_id == bot_id: - return bot - - raise UnknownBotError(bot_id=bot_id) - - def get_token_for_bot(self, bot_id: UUID) -> str: - """Search token in bot saved tokens by bot_id. - - Arguments: - bot_id: UUID of bot for which token should be searched. - - Returns: - Found bot's token. - - Raises: - TokenError: raised of there is not token for bot. - """ - account = self.get_account_by_bot_id(bot_id) - if account.token is not None: - return account.token - - raise TokenError(message_template="Token is empty") diff --git a/botx/bots/mixins/collecting/__init__.py b/botx/bots/mixins/collecting/__init__.py deleted file mode 100644 index a309329c..00000000 --- a/botx/bots/mixins/collecting/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for collecting behaviour mixins.""" diff --git a/botx/bots/mixins/collecting/add_handler.py b/botx/bots/mixins/collecting/add_handler.py deleted file mode 100644 index 4e61e862..00000000 --- a/botx/bots/mixins/collecting/add_handler.py +++ /dev/null @@ -1,51 +0,0 @@ -"""Mixin that defines handler decorator.""" - -from typing import Callable, Optional, Sequence, Union - -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends - - -class AddHandlerMixin: - """Mixin that defines handler decorator.""" - - collector: Collector - - def add_handler( # noqa: WPS211 - self, - handler: Callable, - *, - body: Optional[str] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[Depends]] = None, - ) -> None: - """Create new handler from passed arguments and store it inside. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - body: body template that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - """ - self.collector.add_handler( - body=body, - handler=handler, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - ) diff --git a/botx/bots/mixins/collecting/default.py b/botx/bots/mixins/collecting/default.py deleted file mode 100644 index 60899dac..00000000 --- a/botx/bots/mixins/collecting/default.py +++ /dev/null @@ -1,60 +0,0 @@ -"""Mixin that defines handler decorator.""" - -from typing import Any, Callable, Optional, Sequence, Union - -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends - - -class DefaultHandlerMixin: - """Mixin that defines handler decorator.""" - - collector: Collector - - def default( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = False, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Add new handler to bot and register it as default handler. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.default( - handler=handler, - command=command, - commands=commands, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/bots/mixins/collecting/handler.py b/botx/bots/mixins/collecting/handler.py deleted file mode 100644 index c793646d..00000000 --- a/botx/bots/mixins/collecting/handler.py +++ /dev/null @@ -1,60 +0,0 @@ -"""Mixin that defines handler decorator.""" - -from typing import Any, Callable, Optional, Sequence, Union - -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends - - -class HandlerMixin: - """Mixin that defines handler decorator.""" - - collector: Collector - - def handler( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Add new handler to bot. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.handler( - handler=handler, - command=command, - commands=commands, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/bots/mixins/collecting/hidden.py b/botx/bots/mixins/collecting/hidden.py deleted file mode 100644 index bc38b489..00000000 --- a/botx/bots/mixins/collecting/hidden.py +++ /dev/null @@ -1,45 +0,0 @@ -"""Mixin that defines handler decorator.""" - -from typing import Any, Callable, Optional, Sequence - -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends - - -class HiddenHandlerMixin: - """Mixin that defines handler decorator.""" - - collector: Collector - - def hidden( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register hidden handler that won't be showed in menu. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.hidden( - handler=handler, - command=command, - commands=commands, - name=name, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/bots/mixins/collecting/system_events.py b/botx/bots/mixins/collecting/system_events.py deleted file mode 100644 index 59c8b3af..00000000 --- a/botx/bots/mixins/collecting/system_events.py +++ /dev/null @@ -1,238 +0,0 @@ -"""Mixin that defines handler decorator.""" - -from typing import Any, Callable, Optional, Sequence - -from botx.collecting.collectors.collector import Collector -from botx.dependencies.models import Depends -from botx.models.enums import SystemEvents - - -class SystemEventsHandlerMixin: # noqa: WPS214 - """Mixin that defines handler decorator.""" - - collector: Collector - - def system_event( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - event: Optional[SystemEvents] = None, - events: Optional[Sequence[SystemEvents]] = None, - name: Optional[str] = None, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for system event. - - Arguments: - handler: callable that will be used for executing handler. - event: event for triggering this handler. - events: a sequence of events that will trigger handler. - name: optional name for handler that will be used in generating body. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.system_event( - handler=handler, - event=event, - events=events, - name=name, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def chat_created( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `system:chat_created` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.chat_created( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def file_transfer( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `file_transfer` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.file_transfer( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def added_to_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `added_to_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.added_to_chat( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def deleted_from_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `deleted_from_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.deleted_from_chat( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def left_from_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `left_from_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.left_from_chat( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def cts_login( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `cts_login` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.cts_login( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def cts_logout( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `cts_logout` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.cts_logout( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def smartapp_event( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `smartapp_event` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.collector.smartapp_event( - handler=handler, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/bots/mixins/collectors.py b/botx/bots/mixins/collectors.py deleted file mode 100644 index 9cd14774..00000000 --- a/botx/bots/mixins/collectors.py +++ /dev/null @@ -1,76 +0,0 @@ -"""Definition for bot's collecting component. - -All of this methods are just wrappers around inner collector. -""" - -from typing import Any, List, Optional, Sequence - -from botx.bots.mixins.collecting.add_handler import AddHandlerMixin -from botx.bots.mixins.collecting.default import DefaultHandlerMixin -from botx.bots.mixins.collecting.handler import HandlerMixin -from botx.bots.mixins.collecting.hidden import HiddenHandlerMixin -from botx.bots.mixins.collecting.system_events import SystemEventsHandlerMixin -from botx.collecting.collectors.collector import Collector -from botx.collecting.handlers.handler import Handler -from botx.dependencies import models as deps - - -class BotCollectingMixin( # noqa: WPS215 - AddHandlerMixin, - HandlerMixin, - DefaultHandlerMixin, - HiddenHandlerMixin, - SystemEventsHandlerMixin, -): - """Mixin that defines collector-like behaviour.""" - - collector: Collector - - @property - def handlers(self) -> List[Handler]: - """Get handlers registered on this bot. - - Returns: - Registered handlers of bot. - """ - return self.collector.handlers - - def include_collector( - self, - collector: Collector, - *, - dependencies: Optional[Sequence[deps.Depends]] = None, - ) -> None: - """Include handlers from collector into bot. - - Arguments: - collector: collector from which handlers should be copied. - dependencies: optional sequence of dependencies for handlers for this - collector. - """ - self.collector.include_collector(collector, dependencies=dependencies) - - def command_for(self, *args: Any) -> str: - """Find handler and build a command string using passed body query_params. - - Arguments: - args: sequence of elements where first element should be name of handler. - - Returns: - Command string. - """ - return self.collector.command_for(*args) - - def handler_for(self, name: str) -> Handler: - """Find handler in handlers of this bot. - - Find registered handler using using [botx.collector.Collector.handler_for] of - inner collector. - - Arguments: - name: name of handler that should be found. - - Returns: - Handler that was found by name. - """ - return self.collector.handler_for(name) diff --git a/botx/bots/mixins/exceptions.py b/botx/bots/mixins/exceptions.py deleted file mode 100644 index 9be3f9d1..00000000 --- a/botx/bots/mixins/exceptions.py +++ /dev/null @@ -1,46 +0,0 @@ -"""Exception mixin for bot.""" - -from typing import Callable, Type - -from botx import typing -from botx.middlewares.exceptions import ExceptionMiddleware - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class ExceptionHandlersMixin: - """Mixin that defines functions for exception handlers registration.""" - - exception_middleware: ExceptionMiddleware - - def add_exception_handler( - self, - exc_class: Type[Exception], - handler: typing.ExceptionHandler, - ) -> None: - """Register new handler for exception. - - Arguments: - exc_class: exception type that should be handled. - handler: handler for exception. - """ - self.exception_middleware.add_exception_handler(exc_class, handler) - - def exception_handler(self, exc_class: Type[Exception]) -> Callable: - """Register callable as handler for exception. - - Arguments: - exc_class: exception type that should be handled. - - Returns: - Decorator that will register exception and return passed function. - """ - - def decorator(handler: typing.ExceptionHandler) -> Callable: - self.add_exception_handler(exc_class, handler) - return handler - - return decorator diff --git a/botx/bots/mixins/lifespan.py b/botx/bots/mixins/lifespan.py deleted file mode 100644 index 66db16c5..00000000 --- a/botx/bots/mixins/lifespan.py +++ /dev/null @@ -1,50 +0,0 @@ -"""Lifespan mixin for bot.""" - -import asyncio -from typing import List -from weakref import WeakSet - -from botx.concurrency import callable_to_coroutine -from botx.typing import BotLifespanEvent - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class LifespanMixin: - """Lifespan events mixin for bot.""" - - #: currently running tasks. - tasks: WeakSet - - #: startup events. - startup_events: List[BotLifespanEvent] - - #: shutdown events. - shutdown_events: List[BotLifespanEvent] - - async def start(self) -> None: - """Run all startup events and other initialization stuff.""" - for event in self.startup_events: - await callable_to_coroutine(event, self) - - async def shutdown(self) -> None: - """Wait for all running handlers shutdown.""" - await self.wait_current_handlers() - - for event in self.shutdown_events: - await callable_to_coroutine(event, self) - - async def wait_current_handlers(self) -> None: - """Wait until all current tasks are done.""" - if self.tasks: - tasks, _ = await asyncio.wait( - self.tasks, - return_when=asyncio.ALL_COMPLETED, - ) - for task in tasks: - task.result() - - self.tasks.clear() diff --git a/botx/bots/mixins/middlewares.py b/botx/bots/mixins/middlewares.py deleted file mode 100644 index a8cb317d..00000000 --- a/botx/bots/mixins/middlewares.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Implementation for bot classes.""" - -from typing import Any, Callable, Type - -from botx import typing -from botx.middlewares.base import BaseMiddleware -from botx.middlewares.exceptions import ExceptionMiddleware - - -class MiddlewareMixin: - """Middleware mixin for bot.""" - - exception_middleware: ExceptionMiddleware - - def add_middleware( - self, - middleware_class: Type[BaseMiddleware], - **kwargs: Any, - ) -> None: - """Register new middleware for execution before handler. - - Arguments: - middleware_class: middleware that should be registered. - kwargs: arguments that are required for middleware initialization. - """ - self.exception_middleware.executor = middleware_class( - self.exception_middleware.executor, - **kwargs, - ) - - def middleware(self, handler: typing.Executor) -> Callable: - """Register callable as middleware for request. - - Arguments: - handler: handler for middleware logic. - - Returns: - Passed `handler` callable. - """ - self.add_middleware(BaseMiddleware, dispatch=handler) - return handler diff --git a/botx/bots/mixins/requests/__init__.py b/botx/bots/mixins/requests/__init__.py deleted file mode 100644 index bf1d0b23..00000000 --- a/botx/bots/mixins/requests/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Mixins for shortcuts for requests to BotX API.""" diff --git a/botx/bots/mixins/requests/bots.py b/botx/bots/mixins/requests/bots.py deleted file mode 100644 index 2bfc5987..00000000 --- a/botx/bots/mixins/requests/bots.py +++ /dev/null @@ -1,32 +0,0 @@ -"""Mixin for shortcut for bots resource requests.""" - -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v2.bots.token import Token - - -class BotsRequestsMixin: - """Mixin for shortcut for bots resource requests.""" - - async def get_token( - self: BotXMethodCallProtocol, - host: str, - bot_id: UUID, - signature: str, - ) -> str: - """Obtain token for bot. - - Arguments: - host: host on which request should be made. - bot_id: ID of bot for which token should be obtained. - signature: calculated signature of bot. - - Returns: - Obtained token. - """ - return await self.call_method( - Token(bot_id=bot_id, signature=signature), - host=host, - bot_id=bot_id, - ) diff --git a/botx/bots/mixins/requests/call_protocol.py b/botx/bots/mixins/requests/call_protocol.py deleted file mode 100644 index 54d7a296..00000000 --- a/botx/bots/mixins/requests/call_protocol.py +++ /dev/null @@ -1,26 +0,0 @@ -"""Protocol for using in mixins to make mypy and typing happy.""" -from typing import Any, Optional -from uuid import UUID - -from botx.clients.methods.base import BotXMethod -from botx.models.messages.sending.credentials import SendingCredentials - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class BotXMethodCallProtocol(Protocol): - """Protocol for using in mixins to make mypy and typing happy.""" - - async def call_method( # noqa: WPS211 - self, - method: BotXMethod[Any], - *, - host: Optional[str] = None, - token: Optional[str] = None, - bot_id: Optional[UUID] = None, - credentials: Optional[SendingCredentials] = None, - ) -> Any: - """Send request to BotX API through bot's async client.""" diff --git a/botx/bots/mixins/requests/chats.py b/botx/bots/mixins/requests/chats.py deleted file mode 100644 index 4fa3bf0f..00000000 --- a/botx/bots/mixins/requests/chats.py +++ /dev/null @@ -1,226 +0,0 @@ -"""Mixin for shortcut for chats resource requests.""" - -from typing import List, Optional -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.chats.add_admin_role import AddAdminRole -from botx.clients.methods.v3.chats.add_user import AddUser -from botx.clients.methods.v3.chats.chat_list import ChatList -from botx.clients.methods.v3.chats.create import Create -from botx.clients.methods.v3.chats.info import Info -from botx.clients.methods.v3.chats.pin_message import PinMessage -from botx.clients.methods.v3.chats.remove_user import RemoveUser -from botx.clients.methods.v3.chats.stealth_disable import StealthDisable -from botx.clients.methods.v3.chats.stealth_set import StealthSet -from botx.clients.methods.v3.chats.unpin_message import UnpinMessage -from botx.models.chats import BotChatList, ChatFromSearch -from botx.models.enums import ChatTypes -from botx.models.messages.sending.credentials import SendingCredentials - - -class ChatsRequestsMixin: - """Mixin for shortcut for chats resource requests.""" - - async def create_chat( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - name: str, - members: List[UUID], - chat_type: ChatTypes, - shared_history: bool = False, - description: Optional[str] = None, - avatar: Optional[str] = None, - ) -> UUID: - """Create new chat. - - Arguments: - credentials: credentials for making request. - name: name of chat that should be created. - members: HUIDs of users that should be added into chat. - chat_type: chat type. - description: description of new chat. - avatar: logo image of chat. - shared_history: chat history is available to newcomers. - - Returns: - ID of created chat. - """ - return await self.call_method( - Create( - name=name, - description=description, - members=members, - avatar=avatar, - chat_type=chat_type, - shared_history=shared_history, - ), - credentials=credentials, - ) - - async def get_chat_info( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - ) -> ChatFromSearch: - """Return chat's info. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat for about which information should be retrieving. - - Returns: - Information about chat. - """ - return await self.call_method( - Info(group_chat_id=chat_id), - credentials=credentials, - ) - - async def get_bot_chats( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - ) -> BotChatList: - """Return list of bot's chats. - - Arguments: - credentials: credentials for making request. - - Returns: - List of bot's chats. - """ - return await self.call_method( - ChatList(), - credentials=credentials, - ) - - async def enable_stealth_mode( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - disable_web: bool = False, - burn_in: Optional[int] = None, - expire_in: Optional[int] = None, - ) -> None: - """Enable stealth mode. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat where stealth should be enabled. - disable_web: should messages be shown in web. - burn_in: time of messages burning after read. - expire_in: time of messages burning after send. - """ - await self.call_method( - StealthSet( - group_chat_id=chat_id, - disable_web=disable_web, - burn_in=burn_in, - expire_in=expire_in, - ), - credentials=credentials, - ) - - async def disable_stealth_mode( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - ) -> None: - """Disable stealth mode. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat where stealth should be disabled. - """ - await self.call_method( - StealthDisable(group_chat_id=chat_id), - credentials=credentials, - ) - - async def add_users( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - user_huids: List[UUID], - ) -> None: - """Add users to chat. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat into which users should be added. - user_huids: IDs of users that should be added into chat. - """ - await self.call_method( - AddUser(group_chat_id=chat_id, user_huids=user_huids), - credentials=credentials, - ) - - async def remove_users( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - user_huids: List[UUID], - ) -> None: - """Remove users from chat. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat from which users should be removed. - user_huids: HUID of users that should be removed. - """ - await self.call_method( - RemoveUser(group_chat_id=chat_id, user_huids=user_huids), - credentials=credentials, - ) - - async def add_admin_roles( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - user_huids: List[UUID], - ) -> None: - """Promote users in chat to admins. - - Arguments: - credentials: credentials for making request. - chat_id: UUID of chat where action should be performed. - user_huids: HUIDs of users that should be promoted to admins. - """ - await self.call_method( - AddAdminRole(group_chat_id=chat_id, user_huids=user_huids), - credentials=credentials, - ) - - async def pin_message( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - sync_id: UUID, - ) -> None: - """Pin message in chat. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat where message should be pinned. - sync_id: ID of message that should be pinned. - """ - await self.call_method( - PinMessage(chat_id=chat_id, sync_id=sync_id), - credentials=credentials, - ) - - async def unpin_message( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - chat_id: UUID, - ) -> None: - """Unpin message in chat. - - Arguments: - credentials: credentials for making request. - chat_id: ID of chat where message should be unpinned. - """ - await self.call_method( - UnpinMessage(chat_id=chat_id), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/command.py b/botx/bots/mixins/requests/command.py deleted file mode 100644 index 3f28cc12..00000000 --- a/botx/bots/mixins/requests/command.py +++ /dev/null @@ -1,54 +0,0 @@ -"""Mixin for shortcut for command resource requests.""" - -from typing import cast -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.options import ResultPayloadOptions -from botx.models.messages.sending.payload import MessagePayload - - -class CommandRequestsMixin: - """Mixin for shortcut for command resource requests.""" - - async def send_command_result( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - payload: MessagePayload, - ) -> UUID: - """Send command result into chat. - - Arguments: - credentials: credentials for making request. - payload: payload for command result. - - Returns: - ID sent message. - """ - return await self.call_method( - CommandResult( - sync_id=cast(UUID, credentials.sync_id), - event_sync_id=credentials.message_id, - result=ResultPayload( - body=payload.text, - metadata=payload.metadata, - bubble=payload.markup.bubbles, - keyboard=payload.markup.keyboard, - mentions=payload.options.mentions, - opts=ResultPayloadOptions( - silent_response=payload.options.silent_response, - ), - ), - recipients=payload.options.recipients, - file=payload.file, - opts=ResultOptions( - stealth_mode=payload.options.stealth_mode, - notification_opts=payload.options.notifications, - ), - ), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/events.py b/botx/bots/mixins/requests/events.py deleted file mode 100644 index a56ff789..00000000 --- a/botx/bots/mixins/requests/events.py +++ /dev/null @@ -1,94 +0,0 @@ -"""Mixin for shortcut for events resource requests.""" -from typing import List, Optional -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.events.edit_event import EditEvent -from botx.clients.methods.v3.events.reply_event import ReplyEvent -from botx.clients.types.message_payload import ResultPayload, UpdatePayload -from botx.clients.types.options import ResultOptions -from botx.models.entities import Mention -from botx.models.messages.message import File -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.markup import MessageMarkup -from botx.models.messages.sending.payload import UpdatePayload as SendingUpdatePayload - - -class EventsRequestsMixin: - """Mixin that defines methods for communicating with BotX API.""" - - async def update_message( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - update: SendingUpdatePayload, - ) -> None: - """Change message by it's event id. - - Arguments: - credentials: credentials that are used for sending message. *sync_id* is - required for credentials. - update: update of message content. - - Raises: - ValueError: raised if sync_id wasn't provided - """ - if not credentials.sync_id: - raise ValueError("sync_id is required for message update") - - await self.call_method( - EditEvent( - sync_id=credentials.sync_id, - result=UpdatePayload( - body=update.text, - metadata=update.metadata, - keyboard=update.keyboard, - bubble=update.bubbles, - mentions=update.mentions, - ), - file=update.file, - ), - credentials=credentials, - ) - - async def reply( # noqa: WPS211 - self: BotXMethodCallProtocol, - source_sync_id: UUID, - credentials: SendingCredentials, - text: str = "", - *, - file: Optional[File] = None, - markup: Optional[MessageMarkup] = None, - mentions: Optional[List[Mention]] = None, - opts: Optional[ResultOptions] = None, - ) -> None: - """Reply on message by source_sync_id. - - Arguments: - text: text of message. - source_sync_id: source message uuid that will replied. - file: attachment file. - markup: markup of sending message. - opts: options of sending message. - credentials: credentials for making request. - mentions: mentions in message. - - Raises: - ValueError: empty text. - """ - if not (text or file or mentions): - raise ValueError("text or file or mention required") - - await self.call_method( - ReplyEvent( - source_sync_id=source_sync_id, - result=ResultPayload( - body=text, - keyboard=markup.keyboard if markup else [], - bubble=markup.bubbles if markup else [], - mentions=mentions or [], - ), - file=file, - opts=opts or ResultOptions(), - ), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/files.py b/botx/bots/mixins/requests/files.py deleted file mode 100644 index 50c1678e..00000000 --- a/botx/bots/mixins/requests/files.py +++ /dev/null @@ -1,87 +0,0 @@ -"""Mixin for shortcut for files resource requests.""" - -from typing import Optional -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.clients.async_client import AsyncClient -from botx.clients.methods.v3.files.download import DownloadFile -from botx.clients.methods.v3.files.upload import UploadFile -from botx.clients.types.upload_file import UploadingFileMeta -from botx.models.files import File, MetaFile -from botx.models.messages.sending.credentials import SendingCredentials - - -class FilesRequestsMixin: - """Mixin for shortcut for files resource requests.""" - - client: AsyncClient - - async def upload_file( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - sending_file: File, - group_chat_id: UUID, - *, - duration: Optional[int] = None, - caption: Optional[str] = None, - ) -> MetaFile: - """Upload file to the chat. - - Arguments: - credentials: credentials for making request. - sending_file: file to upload. - group_chat_id: ID of the chat that accepts the file. - duration: duration of the voice or the video. - caption: file caption. - - Returns: - File metadata. - """ - return await self.call_method( - UploadFile( - group_chat_id=group_chat_id, - file=sending_file, - meta=UploadingFileMeta( - duration=duration, - caption=caption, - ), - ), - credentials=credentials, - ) - - async def download_file( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - file_id: UUID, - group_chat_id: UUID, - *, - file_name: Optional[str] = None, - is_preview: bool = False, - ) -> File: - """Download file from the chat. - - Arguments: - credentials: credentials for making request. - file_id: ID of the file. - group_chat_id: ID of the chat that accepts the file. - file_name: file name to be assigned instead of default name. - is_preview: get preview or file. - - Returns: - Downloaded file. - """ - file = await self.call_method( - DownloadFile( - file_id=file_id, - group_chat_id=group_chat_id, - is_preview=is_preview, - ), - credentials=credentials, - ) - - if file_name: - ext = file.file_name.split(".", maxsplit=1)[1] - file.file_name = "{name}.{ext}".format(name=file_name, ext=ext) - - return file diff --git a/botx/bots/mixins/requests/internal_bot_notification.py b/botx/bots/mixins/requests/internal_bot_notification.py deleted file mode 100644 index 9a78f112..00000000 --- a/botx/bots/mixins/requests/internal_bot_notification.py +++ /dev/null @@ -1,47 +0,0 @@ -"""Mixin for shortcut for internal bot notification resource requests.""" - -from typing import Any, Dict, List, Optional -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.clients.types.message_payload import InternalBotNotificationPayload -from botx.models.messages.sending.credentials import SendingCredentials - - -class InternalBotNotificationRequestsMixin: - """Mixin for shortcut for internal bot notification resource requests.""" - - async def internal_bot_notification( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - group_chat_id: UUID, - text: str, - sender: Optional[str] = None, - recipients: Optional[List[UUID]] = None, - opts: Optional[Dict[str, Any]] = None, - ) -> UUID: - """Send internal bot notifications into chat. - - Arguments: - credentials: credentials for making request. - group_chat_id: ID of chats into which message should be sent. - text: notification text. - sender: information about notification sender. - recipients: List of recipients' UUIDs (send to all if None) - opts: additional user-defined data to send - - Returns: - Sync ID of sent notification. - """ - return await self.call_method( - InternalBotNotification( - group_chat_id=group_chat_id, - recipients=recipients, - data=InternalBotNotificationPayload(message=text, sender=sender), - opts=opts or {}, - ), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/mixin.py b/botx/bots/mixins/requests/mixin.py deleted file mode 100644 index 95d2015e..00000000 --- a/botx/bots/mixins/requests/mixin.py +++ /dev/null @@ -1,105 +0,0 @@ -"""Definition for mixin that defines BotX API methods.""" -from typing import Optional, TypeVar, cast -from uuid import UUID - -from loguru import logger - -from botx.bots.mixins.requests import bots # noqa: WPS235 -from botx.bots.mixins.requests import ( - chats, - command, - events, - files, - internal_bot_notification, - notification, - smartapps, - stickers, - users, -) -from botx.clients.clients.async_client import AsyncClient -from botx.clients.methods.base import BotXMethod -from botx.models.credentials import BotXCredentials -from botx.models.messages.sending.credentials import SendingCredentials - -ResponseT = TypeVar("ResponseT") - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class CredentialsSearchProtocol(Protocol): - """Protocol for search token in local credentials.""" - - def get_token_for_bot(self, bot_id: UUID) -> str: - """Search token in local credentials.""" - - def get_account_by_bot_id(self, bot_id: UUID) -> BotXCredentials: - """Get bot credentials by bot id.""" - - -# A lot of base classes since it's mixin for all shorthands for BotX API requests -class BotXRequestsMixin( # noqa: WPS215 - bots.BotsRequestsMixin, - chats.ChatsRequestsMixin, - command.CommandRequestsMixin, - events.EventsRequestsMixin, - notification.NotificationRequestsMixin, - users.UsersRequestsMixin, - internal_bot_notification.InternalBotNotificationRequestsMixin, - files.FilesRequestsMixin, - smartapps.SmartAppMixin, - stickers.StickersMixin, -): - """Mixin that defines methods for communicating with BotX API.""" - - client: AsyncClient - - # TODO: remove SendingCredential from client module. - async def call_method( # noqa: WPS211 - self, - method: BotXMethod[ResponseT], - *, - host: Optional[str] = None, - token: Optional[str] = None, - bot_id: Optional[UUID] = None, - credentials: Optional[SendingCredentials] = None, - ) -> ResponseT: - """Call method with async client. - - Arguments: - method: method that should be user for request. - host: host where request should be sent. - token: token for method. - bot_id: ID of bot that send request. - credentials: credentials for making request. - - Returns: - Response for method. - """ - if credentials is not None: - debug_bot_id = credentials.bot_id - host = cast(str, credentials.host) - bot_id = cast(UUID, credentials.bot_id) - method.configure( - host=host, - token=cast(CredentialsSearchProtocol, self).get_token_for_bot(bot_id), - ) - else: - debug_bot_id = bot_id - method.configure(host=host or method.host, token=token or method.token) - - request = self.client.build_request(method) - - logger.bind( - botx_client=True, - payload=request.dict(exclude={"expected_type"}), - ).debug( - "send {0} request to bot {1}", - method.__repr_name__(), # noqa: WPS609 - debug_bot_id, - ) - - response = await self.client.execute(request) - return await self.client.process_response(method, response) diff --git a/botx/bots/mixins/requests/notification.py b/botx/bots/mixins/requests/notification.py deleted file mode 100644 index a4dfe1e9..00000000 --- a/botx/bots/mixins/requests/notification.py +++ /dev/null @@ -1,107 +0,0 @@ -"""Mixin for shortcut for notification resource requests.""" - -from typing import Optional, Sequence -from uuid import UUID - -from botx import converters -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v3.notification.notification import Notification -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.options import ResultPayloadOptions -from botx.models.messages.sending.payload import MessagePayload - - -class NotificationRequestsMixin: - """Mixin for shortcut for notification resource requests.""" - - async def send_notification( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - payload: MessagePayload, - group_chat_ids: Optional[Sequence[UUID]] = None, - ) -> None: - """Send notifications into chat. - - Arguments: - credentials: credentials for making request. - payload: payload for notification. - group_chat_ids: IDS of chats into which message should be sent. - """ - if group_chat_ids is not None: - chat_ids = converters.optional_sequence_to_list(group_chat_ids) - elif credentials.chat_id: - chat_ids = [credentials.chat_id] - else: - chat_ids = [] - - await self.call_method( - Notification( - group_chat_ids=chat_ids, - result=ResultPayload( - body=payload.text, - metadata=payload.metadata, - bubble=payload.markup.bubbles, - keyboard=payload.markup.keyboard, - mentions=payload.options.mentions, - opts=ResultPayloadOptions( - silent_response=payload.options.silent_response, - ), - ), - recipients=payload.options.recipients, - file=payload.file, - opts=ResultOptions( - stealth_mode=payload.options.stealth_mode, - raw_mentions=payload.options.raw_mentions, - notification_opts=payload.options.notifications, - ), - ), - credentials=credentials, - ) - - async def send_direct_notification( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - payload: MessagePayload, - ) -> UUID: - """Send notification into chat. - - Arguments: - credentials: credentials for making request. - payload: payload for notification. - - Returns: - ID sent message. - - Raises: - ValueError: raised if chat_id wasn't provided - """ - if not credentials.chat_id: - raise ValueError("chat_id is required to send direct notification") - - return await self.call_method( - NotificationDirect( - group_chat_id=credentials.chat_id, - event_sync_id=credentials.message_id, - result=ResultPayload( - body=payload.text, - metadata=payload.metadata, - bubble=payload.markup.bubbles, - keyboard=payload.markup.keyboard, - mentions=payload.options.mentions, - opts=ResultPayloadOptions( - silent_response=payload.options.silent_response, - ), - ), - recipients=payload.options.recipients, - file=payload.file, - opts=ResultOptions( - stealth_mode=payload.options.stealth_mode, - raw_mentions=payload.options.raw_mentions, - notification_opts=payload.options.notifications, - ), - ), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/smartapps.py b/botx/bots/mixins/requests/smartapps.py deleted file mode 100644 index 69e2ac27..00000000 --- a/botx/bots/mixins/requests/smartapps.py +++ /dev/null @@ -1,62 +0,0 @@ -"""Mixin for shortcut for smartapp.""" - -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.smartapps.smartapp_event import SmartAppEvent -from botx.clients.methods.v3.smartapps.smartapp_notification import SmartAppNotification -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.smartapps import SendingSmartAppEvent, SendingSmartAppNotification - - -class SmartAppMixin: - """Mixin for shortcut for smartapp methods.""" - - async def send_smartapp_event( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - smartapp_event: SendingSmartAppEvent, - ) -> UUID: - """Send smartapp event into chat. - - Arguments: - credentials: credentials for making request. - smartapp_event: SmartpApp event. - - Returns: - ID sent message. - """ - return await self.call_method( - SmartAppEvent( - ref=smartapp_event.ref, - smartapp_id=smartapp_event.smartapp_id, - data=smartapp_event.data, - opts=smartapp_event.opts, - smartapp_api_version=smartapp_event.smartapp_api_version, - group_chat_id=smartapp_event.group_chat_id, - files=smartapp_event.files, - async_files=smartapp_event.async_files, - ), - credentials=credentials, - ) - - async def send_smartapp_notification( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - smartapp_notification: SendingSmartAppNotification, - ) -> None: - """Send smartapp notification into chat. - - Arguments: - credentials: credentials for making request. - smartapp_notification: Smartapp notification. - """ - await self.call_method( - SmartAppNotification( - group_chat_id=smartapp_notification.group_chat_id, - smartapp_counter=smartapp_notification.smartapp_counter, - opts=smartapp_notification.opts, - smartapp_api_version=smartapp_notification.smartapp_api_version, - ), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/stickers.py b/botx/bots/mixins/requests/stickers.py deleted file mode 100644 index 8ca02581..00000000 --- a/botx/bots/mixins/requests/stickers.py +++ /dev/null @@ -1,199 +0,0 @@ -"""Mixin for shortcut for users resource requests.""" - -from typing import List, Optional, Tuple -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.stickers.add_sticker import AddSticker -from botx.clients.methods.v3.stickers.create_sticker_pack import CreateStickerPack -from botx.clients.methods.v3.stickers.delete_sticker import DeleteSticker -from botx.clients.methods.v3.stickers.delete_sticker_pack import DeleteStickerPack -from botx.clients.methods.v3.stickers.edit_sticker_pack import EditStickerPack -from botx.clients.methods.v3.stickers.sticker import GetSticker -from botx.clients.methods.v3.stickers.sticker_pack import GetStickerPack -from botx.clients.methods.v3.stickers.sticker_pack_list import GetStickerPackList -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.stickers import ( - Sticker, - StickerFromPack, - StickerPack, - StickerPackList, - StickerPackPreview, -) - - -class StickersMixin: # noqa: WPS214 - """Mixin for shortcut for users resource requests.""" - - async def get_sticker_pack_list( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - user_huid: Optional[UUID] = None, - limit: int = 1, - after: Optional[str] = None, - ) -> Tuple[List[StickerPackPreview], Optional[str]]: - """Get sticker pack list. - - Arguments: - credentials: credentials for making request. - user_huid: author HUID. - limit: returning value count. - after: cursor hash for pagination. - - Returns: - Sticker packs list and cursor. - """ - response: StickerPackList = await self.call_method( - GetStickerPackList(user_huid=user_huid, limit=limit, after=after), - credentials=credentials, - ) - - return response.packs, response.pagination.after - - async def get_sticker_pack( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - ) -> StickerPack: - """Get sticker pack. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - - Returns: - StickerPack entity. - """ - return await self.call_method( - GetStickerPack(pack_id=pack_id), - credentials=credentials, - ) - - async def get_sticker_from_pack( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - sticker_id: UUID, - ) -> StickerFromPack: - """Get sticker from pack. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - sticker_id: sticker ID. - - Returns: - StickerFromPack entity. - """ - return await self.call_method( - GetSticker(pack_id=pack_id, sticker_id=sticker_id), - credentials=credentials, - ) - - async def create_sticker_pack( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - name: str, - user_huid: UUID, - ) -> StickerPack: - """Create sticker pack. - - Arguments: - credentials: credentials for making request. - name: sticker pack name. - user_huid: author HUID. - - Returns: - StickerPackPreview entity. - """ - return await self.call_method( - CreateStickerPack(name=name, user_huid=user_huid), - credentials=credentials, - ) - - async def add_sticker( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - emoji: str, - image: str, - ) -> Sticker: - """Add sticker. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - emoji: emoji that the sticker will be associated with. - image: sticker image. - - Returns: - Sticker entity. - """ - return await self.call_method( - AddSticker(pack_id=pack_id, emoji=emoji, image=image), - credentials=credentials, - ) - - async def edit_sticker_pack( # noqa: WPS211 - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - name: str, - preview: Optional[UUID] = None, - stickers_order: Optional[List[UUID]] = None, - ) -> StickerPack: - """Edit sticker pack. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - name: sticker pack name. - preview: sticker pack preview. - stickers_order: stickers order in sticker pack. - - Returns: - StickerPack entity. - """ - return await self.call_method( - EditStickerPack( - pack_id=pack_id, - name=name, - preview=preview, - stickers_order=stickers_order, - ), - credentials=credentials, - ) - - async def delete_sticker_pack( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - ) -> None: - """Delete sticker pack. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - """ - await self.call_method( - DeleteStickerPack(pack_id=pack_id), - credentials=credentials, - ) - - async def delete_sticker( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - pack_id: UUID, - sticker_id: UUID, - ) -> None: - """Delete sticker. - - Arguments: - credentials: credentials for making request. - pack_id: sticker pack ID. - sticker_id: sticker ID. - """ - await self.call_method( - DeleteSticker(pack_id=pack_id, sticker_id=sticker_id), - credentials=credentials, - ) diff --git a/botx/bots/mixins/requests/users.py b/botx/bots/mixins/requests/users.py deleted file mode 100644 index 8cc24569..00000000 --- a/botx/bots/mixins/requests/users.py +++ /dev/null @@ -1,52 +0,0 @@ -"""Mixin for shortcut for users resource requests.""" - -from typing import Optional, Tuple -from uuid import UUID - -from botx.bots.mixins.requests.call_protocol import BotXMethodCallProtocol -from botx.clients.methods.v3.users.by_email import ByEmail -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.clients.methods.v3.users.by_login import ByLogin -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.users import UserFromSearch - - -class UsersRequestsMixin: - """Mixin for shortcut for users resource requests.""" - - async def search_user( - self: BotXMethodCallProtocol, - credentials: SendingCredentials, - *, - user_huid: Optional[UUID] = None, - email: Optional[str] = None, - ad: Optional[Tuple[str, str]] = None, - ) -> UserFromSearch: - """Search user by one of provided params for search. - - Arguments: - credentials: credentials for making request. - user_huid: HUID of user. - email: email of user. - ad: AD login and domain of user. - - Returns: - Information about user. - - Raises: - ValueError: raised if none of provided params were filled. - """ - if user_huid is not None: - return await self.call_method( - ByHUID(user_huid=user_huid), - credentials=credentials, - ) - elif email is not None: - return await self.call_method(ByEmail(email=email), credentials=credentials) - elif ad is not None: - return await self.call_method( - ByLogin(ad_login=ad[0], ad_domain=ad[1]), - credentials=credentials, - ) - - raise ValueError("one of user_huid, email or ad query_params should be filled") diff --git a/botx/bots/mixins/sending.py b/botx/bots/mixins/sending.py deleted file mode 100644 index da2d93ff..00000000 --- a/botx/bots/mixins/sending.py +++ /dev/null @@ -1,175 +0,0 @@ -"""Definition for mixin that defines helpers for sending message.""" - -from typing import Any, BinaryIO, Dict, Optional, TextIO, Union -from uuid import UUID - -from botx.models.files import File -from botx.models.messages.message import Message -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.markup import MessageMarkup -from botx.models.messages.sending.message import SendingMessage -from botx.models.messages.sending.options import MessageOptions -from botx.models.messages.sending.payload import MessagePayload, UpdatePayload - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class ResultSendProtocol(Protocol): - """Protocol for object that can create new or update message.""" - - async def send_command_result( - self, - credentials: SendingCredentials, - payload: MessagePayload, - ) -> UUID: - """Send command result.""" - - async def send_direct_notification( - self, - credentials: SendingCredentials, - payload: MessagePayload, - ) -> UUID: - """Send notification.""" - - async def update_message( - self, - credentials: SendingCredentials, - update: UpdatePayload, - ) -> None: - """Update existing message.""" - - -class MessageSendProtocol(Protocol): - """Protocol for object that can send complex message.""" - - async def send(self, message: SendingMessage) -> UUID: - """Send message.""" - - -class SendingMixin: - """Mixin that defines helpers for sending messages.""" - - async def send_message( # noqa: WPS211 - self: MessageSendProtocol, - text: str, - credentials: SendingCredentials, - *, - file: Optional[Union[BinaryIO, TextIO]] = None, - markup: Optional[MessageMarkup] = None, - options: Optional[MessageOptions] = None, - ) -> UUID: - """Send message as answer to command or notification to chat and get it id. - - Arguments: - text: text that should be sent to client. - credentials: credentials that are used for sending message. - file: file that should be attached to message. - markup: message markup that should be attached to message. - options: extra options for message. - - Returns: - `UUID` of sent event. - """ - message = SendingMessage( - text=text, - markup=markup, - options=options, - credentials=credentials, - ) - if file: - message.add_file(file) - - return await self.send(message) - - async def send( - self: ResultSendProtocol, - message: SendingMessage, - *, - update: bool = False, - ) -> UUID: - """Send message as direct notification to chat and get it id. - - Arguments: - message: message that should be sent to chat. - update: if True then check, that `message_id` was set in credentials and - update existing message with this ID. - - Returns: - `UUID` of sent event. - """ - if message.credentials.message_id is not None and update: - await self.update_message( - message.credentials.copy( - update={"sync_id": message.credentials.message_id}, - ), - UpdatePayload.from_sending_payload(message.payload), - ) - return message.credentials.message_id - - return await self.send_direct_notification(message.credentials, message.payload) - - async def answer_message( # noqa: WPS211 - self: MessageSendProtocol, - text: str, - message: Message, - *, - metadata: Optional[Dict[str, Any]] = None, - file: Optional[Union[BinaryIO, TextIO, File]] = None, - markup: Optional[MessageMarkup] = None, - options: Optional[MessageOptions] = None, - embed_mentions: bool = False, - ) -> UUID: - """Answer on incoming message and return id of new message.. - - !!! warning - This method should be used only in handlers. - - Arguments: - text: text that should be sent in message. - message: incoming message. - file: file that can be attached to the message. - markup: bubbles and keyboard that can be attached to the message. - options: additional message options, like mentions or notifications - configuration. - metadata: dict of message metadata. - embed_mentions: get mentions from text. - - Returns: - `UUID` of sent event. - """ - sending_message = SendingMessage( - text=text, - credentials=message.credentials, - markup=markup, - options=options, - metadata=metadata, - embed_mentions=embed_mentions, - ) - if file: - sending_message.add_file(file) - - return await self.send(sending_message) - - async def send_file( - self: MessageSendProtocol, - file: Union[TextIO, BinaryIO, File], - credentials: SendingCredentials, - filename: Optional[str] = None, - ) -> UUID: - """Send file in chat and return id of message. - - Arguments: - file: file-like object that will be sent to chat. - credentials: credentials of chat where file should be sent. - filename: name for file that will be used if it can not be accessed from - `file` argument. - - Returns: - `UUID` of sent event. - """ - message = SendingMessage(credentials=credentials) - message.add_file(file, filename) - return await self.send(message) diff --git a/docs/src/development/dependencies_injection/__init__.py b/botx/client/__init__.py similarity index 100% rename from docs/src/development/dependencies_injection/__init__.py rename to botx/client/__init__.py diff --git a/botx/client/authorized_botx_method.py b/botx/client/authorized_botx_method.py new file mode 100644 index 00000000..e574617e --- /dev/null +++ b/botx/client/authorized_botx_method.py @@ -0,0 +1,50 @@ +from contextlib import asynccontextmanager +from typing import Any, AsyncGenerator, Dict + +import httpx + +from botx.client.botx_method import BotXMethod, response_exception_thrower +from botx.client.exceptions.common import InvalidBotAccountError +from botx.client.get_token import get_token + + +class AuthorizedBotXMethod(BotXMethod): + status_handlers = {401: response_exception_thrower(InvalidBotAccountError)} + + async def _botx_method_call( + self, + *args: Any, + **kwargs: Any, + ) -> httpx.Response: + headers = kwargs.pop("headers", {}) + await self._add_authorization_headers(headers) + + return await super()._botx_method_call(*args, headers=headers, **kwargs) + + @asynccontextmanager + async def _botx_method_stream( + self, + *args: Any, + **kwargs: Any, + ) -> AsyncGenerator[httpx.Response, None]: + headers = kwargs.pop("headers", {}) + await self._add_authorization_headers(headers) + + async with super()._botx_method_stream( + *args, + headers=headers, + **kwargs, + ) as response: + yield response + + async def _add_authorization_headers(self, headers: Dict[str, Any]) -> None: + token = self._bot_accounts_storage.get_token_or_none(self._bot_id) + if not token: + token = await get_token( + self._bot_id, + self._httpx_client, + self._bot_accounts_storage, + ) + self._bot_accounts_storage.set_token(self._bot_id, token) + + headers.update({"Authorization": f"Bearer {token}"}) diff --git a/docs/src/development/first_steps/__init__.py b/botx/client/bots_api/__init__.py similarity index 100% rename from docs/src/development/first_steps/__init__.py rename to botx/client/bots_api/__init__.py diff --git a/botx/client/bots_api/get_token.py b/botx/client/bots_api/get_token.py new file mode 100644 index 00000000..2c4c2e5a --- /dev/null +++ b/botx/client/bots_api/get_token.py @@ -0,0 +1,42 @@ +from typing import Literal + +from botx.client.botx_method import BotXMethod, response_exception_thrower +from botx.client.exceptions.common import InvalidBotAccountError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIGetTokenRequestPayload(UnverifiedPayloadBaseModel): + signature: str + + @classmethod + def from_domain(cls, signature: str) -> "BotXAPIGetTokenRequestPayload": + return cls(signature=signature) + + +class BotXAPIGetTokenResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: str + + def to_domain(self) -> str: + return self.result + + +class GetTokenMethod(BotXMethod): + status_handlers = {401: response_exception_thrower(InvalidBotAccountError)} + + async def execute( + self, + payload: BotXAPIGetTokenRequestPayload, + ) -> BotXAPIGetTokenResponsePayload: + path = f"/api/v2/botx/bots/{self._bot_id}/token" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPIGetTokenResponsePayload, + response, + ) diff --git a/botx/client/botx_method.py b/botx/client/botx_method.py new file mode 100644 index 00000000..53a5e173 --- /dev/null +++ b/botx/client/botx_method.py @@ -0,0 +1,201 @@ +import json +from contextlib import asynccontextmanager +from json.decoder import JSONDecodeError +from typing import ( + Any, + AsyncGenerator, + Awaitable, + Callable, + Mapping, + NoReturn, + Optional, + Type, + TypeVar, +) +from urllib.parse import urljoin +from uuid import UUID + +import httpx +from mypy_extensions import Arg +from pydantic import ValidationError, parse_obj_as + +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.bot.callbacks_manager import CallbacksManager +from botx.client.exceptions.base import BaseClientError +from botx.client.exceptions.callbacks import BotXMethodFailedCallbackReceivedError +from botx.client.exceptions.http import ( + InvalidBotXResponsePayloadError, + InvalidBotXStatusCodeError, +) +from botx.logger import logger, pformat_jsonable_obj, trim_file_data_in_outgoing_json +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.method_callbacks import BotAPIMethodFailedCallback, BotXMethodCallback + +StatusHandler = Callable[[Arg(httpx.Response, "response")], NoReturn] # noqa: F821 +StatusHandlers = Mapping[int, StatusHandler] + +CallbackExceptionHandler = Callable[ + [Arg(BotAPIMethodFailedCallback, "callback")], # noqa: F821 + NoReturn, +] +ErrorCallbackHandlers = Mapping[str, CallbackExceptionHandler] +TBotXAPIModel = TypeVar("TBotXAPIModel", bound=VerifiedPayloadBaseModel) + + +def response_exception_thrower( + exc: Type[BaseClientError], + comment: Optional[str] = None, +) -> StatusHandler: + def factory(response: httpx.Response) -> NoReturn: + raise exc.from_response(response, comment) + + return factory + + +def callback_exception_thrower( + exc: Type[BaseClientError], + comment: Optional[str] = None, +) -> CallbackExceptionHandler: # noqa: F821 + def factory(callback: BotAPIMethodFailedCallback) -> NoReturn: + raise exc.from_callback(callback, comment) + + return factory + + +class BotXMethod: + status_handlers: StatusHandlers = {} + error_callback_handlers: ErrorCallbackHandlers = {} + + def __init__( + self, + sender_bot_id: UUID, + httpx_client: httpx.AsyncClient, + bot_accounts_storage: BotAccountsStorage, + callbacks_manager: Optional[CallbacksManager] = None, + ) -> None: + self._bot_id = sender_bot_id + self._httpx_client = httpx_client + self._bot_accounts_storage = bot_accounts_storage + self._callbacks_manager = callbacks_manager + + # For MyPy checks + execute: Callable[..., Awaitable[Any]] + + async def execute(self, *args: Any, **kwargs: Any) -> Any: # type: ignore + raise NotImplementedError("You should define `execute` method") + + def _build_url(self, path: str) -> str: + host = self._bot_accounts_storage.get_host(self._bot_id) + return urljoin(f"https://{host}", path) + + def _verify_and_extract_api_model( + self, + model_cls: Type[TBotXAPIModel], + response: httpx.Response, + ) -> TBotXAPIModel: + try: + raw_model = json.loads(response.content) + except JSONDecodeError as decoding_exc: + raise InvalidBotXResponsePayloadError(response) from decoding_exc + + logger.opt(lazy=True).debug( + "Got response from BotX: {json}", + json=lambda: pformat_jsonable_obj(raw_model), + ) + + try: + api_model = parse_obj_as(model_cls, raw_model) + except ValidationError as validation_exc: + raise InvalidBotXResponsePayloadError(response) from validation_exc + + return api_model + + async def _botx_method_call(self, *args: Any, **kwargs: Any) -> httpx.Response: + self._log_outgoing_request(*args, **kwargs) + + response = await self._httpx_client.request(*args, **kwargs) + await self._raise_for_status(response) + + return response + + @asynccontextmanager + async def _botx_method_stream( + self, + *args: Any, + **kwargs: Any, + ) -> AsyncGenerator[httpx.Response, None]: + self._log_outgoing_request(*args, **kwargs) + + async with self._httpx_client.stream(*args, **kwargs) as response: + await self._raise_for_status(response) + yield response + + async def _raise_for_status(self, response: httpx.Response) -> None: + handler = self.status_handlers.get(response.status_code) + if handler: + if not response.is_closed: + await response.aread() + + handler(response) # Handler should raise an exception + + try: + response.raise_for_status() + except httpx.HTTPStatusError as exc: + if not response.is_closed: + await response.aread() + + raise InvalidBotXStatusCodeError(exc.response) + + async def _process_callback( + self, + sync_id: UUID, + wait_callback: bool, + callback_timeout: Optional[int], + ) -> Optional[BotXMethodCallback]: + assert ( + self._callbacks_manager is not None + ), "CallbackManager hasn't been passed to this method" + + self._callbacks_manager.create_botx_method_callback(sync_id) + + if not wait_callback: + return None + + callback = await self._callbacks_manager.wait_botx_method_callback( + sync_id, + callback_timeout, + ) + + if callback.status == "error": + error_handler = self.error_callback_handlers.get(callback.reason) + if not error_handler: + raise BotXMethodFailedCallbackReceivedError(callback) + + error_handler(callback) # Handler should raise an exception + + return callback + + def _log_outgoing_request( + self, + *args: Any, + **kwargs: Any, + ) -> None: + method, url = args + query_params = kwargs.get("params") + json_body = kwargs.get("json") + + log_template = "Performing request to BotX:\n{method} {url}" + if query_params: + log_template += "\nquery: {params}" + if json_body is not None: + log_template += "\njson: {json}" + + logger.opt(lazy=True).debug( + log_template, + method=lambda: method, # If `lazy` enabled, all kwargs should be callable + url=lambda: url, # If `lazy` enabled, all kwargs should be callable + params=lambda: pformat_jsonable_obj(query_params), + json=lambda: pformat_jsonable_obj( + trim_file_data_in_outgoing_json(json_body), + ), + ) diff --git a/docs/src/development/handling_errors/__init__.py b/botx/client/chats_api/__init__.py similarity index 100% rename from docs/src/development/handling_errors/__init__.py rename to botx/client/chats_api/__init__.py diff --git a/botx/client/chats_api/add_admin.py b/botx/client/chats_api/add_admin.py new file mode 100644 index 00000000..a582598b --- /dev/null +++ b/botx/client/chats_api/add_admin.py @@ -0,0 +1,71 @@ +from typing import List, Literal, NoReturn +from uuid import UUID + +import httpx + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.chats import ( + CantUpdatePersonalChatError, + InvalidUsersListError, +) +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.client.exceptions.http import InvalidBotXStatusCodeError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIAddAdminRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + user_huids: List[UUID] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + huids: List[UUID], + ) -> "BotXAPIAddAdminRequestPayload": + return cls(group_chat_id=chat_id, user_huids=huids) + + +class BotXAPIAddAdminResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +def bad_request_error_handler(response: httpx.Response) -> NoReturn: + reason = response.json().get("reason") + + if reason == "chat_members_not_modifiable": + raise CantUpdatePersonalChatError.from_response( + response, + "Personal chat couldn't have admins", + ) + elif reason == "admins_not_changed": + raise InvalidUsersListError.from_response( + response, + "Specified users are already admins or missing from chat", + ) + + raise InvalidBotXStatusCodeError(response) + + +class AddAdminMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 400: bad_request_error_handler, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIAddAdminRequestPayload, + ) -> None: + path = "/api/v3/botx/chats/add_admin" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPIAddAdminResponsePayload, response) diff --git a/botx/client/chats_api/add_user.py b/botx/client/chats_api/add_user.py new file mode 100644 index 00000000..bd39bcf8 --- /dev/null +++ b/botx/client/chats_api/add_user.py @@ -0,0 +1,48 @@ +from typing import List, Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIAddUserRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + user_huids: List[UUID] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + huids: List[UUID], + ) -> "BotXAPIAddUserRequestPayload": + return cls(group_chat_id=chat_id, user_huids=huids) + + +class BotXAPIAddUserResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class AddUserMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIAddUserRequestPayload, + ) -> None: + path = "/api/v3/botx/chats/add_user" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + self._verify_and_extract_api_model( + BotXAPIAddUserResponsePayload, + response, + ) diff --git a/botx/client/chats_api/chat_info.py b/botx/client/chats_api/chat_info.py new file mode 100644 index 00000000..46135d51 --- /dev/null +++ b/botx/client/chats_api/chat_info.py @@ -0,0 +1,88 @@ +from datetime import datetime as dt +from typing import List, Literal, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.chats import ChatInfo, ChatInfoMember +from botx.models.enums import ( + APIChatTypes, + APIUserKinds, + convert_chat_type_to_domain, + convert_user_kind_to_domain, +) + + +class BotXAPIChatInfoRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + + @classmethod + def from_domain(cls, chat_id: UUID) -> "BotXAPIChatInfoRequestPayload": + return cls(group_chat_id=chat_id) + + +class BotXAPIChatInfoMember(VerifiedPayloadBaseModel): + admin: bool + user_huid: UUID + user_kind: APIUserKinds + + +class BotXAPIChatInfoResult(VerifiedPayloadBaseModel): + chat_type: APIChatTypes + creator: UUID + description: Optional[str] = None + group_chat_id: UUID + inserted_at: dt + members: List[BotXAPIChatInfoMember] + name: str + + +class BotXAPIChatInfoResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIChatInfoResult + + def to_domain(self) -> ChatInfo: + members = [ + ChatInfoMember( + is_admin=member.admin, + huid=member.user_huid, + kind=convert_user_kind_to_domain(member.user_kind), + ) + for member in self.result.members + ] + + return ChatInfo( + chat_type=convert_chat_type_to_domain(self.result.chat_type), + creator_id=self.result.creator, + description=self.result.description, + chat_id=self.result.group_chat_id, + created_at=self.result.inserted_at, + members=members, + name=self.result.name, + ) + + +class ChatInfoMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIChatInfoRequestPayload, + ) -> BotXAPIChatInfoResponsePayload: + path = "/api/v3/botx/chats/info" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPIChatInfoResponsePayload, + response, + ) diff --git a/botx/client/chats_api/create_chat.py b/botx/client/chats_api/create_chat.py new file mode 100644 index 00000000..57af0f5f --- /dev/null +++ b/botx/client/chats_api/create_chat.py @@ -0,0 +1,71 @@ +from typing import List, Literal, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.chats import ChatCreationError, ChatCreationProhibitedError +from botx.missing import Missing +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.enums import APIChatTypes, ChatTypes, convert_chat_type_from_domain + + +class BotXAPICreateChatRequestPayload(UnverifiedPayloadBaseModel): + name: str + description: Optional[str] + chat_type: APIChatTypes + members: List[UUID] + shared_history: Missing[bool] + + @classmethod + def from_domain( + cls, + name: str, + chat_type: ChatTypes, + huids: List[UUID], + shared_history: Missing[bool], + description: Optional[str] = None, + ) -> "BotXAPICreateChatRequestPayload": + return cls( + name=name, + chat_type=convert_chat_type_from_domain(chat_type), + members=huids, + description=description, + shared_history=shared_history, + ) + + +class BotXAPIChatIdResult(VerifiedPayloadBaseModel): + chat_id: UUID + + +class BotXAPICreateChatResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIChatIdResult + + def to_domain(self) -> UUID: + return self.result.chat_id + + +class CreateChatMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(ChatCreationProhibitedError), + 422: response_exception_thrower(ChatCreationError), + } + + async def execute( + self, + payload: BotXAPICreateChatRequestPayload, + ) -> BotXAPICreateChatResponsePayload: + path = "/api/v3/botx/chats/create" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPICreateChatResponsePayload, + response, + ) diff --git a/botx/client/chats_api/disable_stealth.py b/botx/client/chats_api/disable_stealth.py new file mode 100644 index 00000000..5a2a1eea --- /dev/null +++ b/botx/client/chats_api/disable_stealth.py @@ -0,0 +1,46 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIDisableStealthRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + + @classmethod + def from_domain( + cls, + chat_id: UUID, + ) -> "BotXAPIDisableStealthRequestPayload": + return cls( + group_chat_id=chat_id, + ) + + +class BotXAPIDisableStealthResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class DisableStealthMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute(self, payload: BotXAPIDisableStealthRequestPayload) -> None: + path = "/api/v3/botx/chats/stealth_disable" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model( + BotXAPIDisableStealthResponsePayload, + response, + ) diff --git a/botx/client/chats_api/list_chats.py b/botx/client/chats_api/list_chats.py new file mode 100644 index 00000000..05db5e0d --- /dev/null +++ b/botx/client/chats_api/list_chats.py @@ -0,0 +1,52 @@ +from datetime import datetime +from typing import List, Literal, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.chats import ChatListItem +from botx.models.enums import APIChatTypes, convert_chat_type_to_domain + + +class BotXAPIListChatResult(VerifiedPayloadBaseModel): + group_chat_id: UUID + chat_type: APIChatTypes + name: str + description: Optional[str] = None + members: List[UUID] + inserted_at: datetime + updated_at: datetime + + +class BotXAPIListChatResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: List[BotXAPIListChatResult] + + def to_domain(self) -> List[ChatListItem]: + return [ + ChatListItem( + chat_id=chat_item.group_chat_id, + chat_type=convert_chat_type_to_domain(chat_item.chat_type), + name=chat_item.name, + description=chat_item.description, + members=chat_item.members, + created_at=chat_item.inserted_at, + updated_at=chat_item.updated_at, + ) + for chat_item in self.result + ] + + +class ListChatsMethod(AuthorizedBotXMethod): + async def execute(self) -> BotXAPIListChatResponsePayload: + path = "/api/v3/botx/chats/list" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + ) + + return self._verify_and_extract_api_model( + BotXAPIListChatResponsePayload, + response, + ) diff --git a/botx/client/chats_api/pin_message.py b/botx/client/chats_api/pin_message.py new file mode 100644 index 00000000..062dc76f --- /dev/null +++ b/botx/client/chats_api/pin_message.py @@ -0,0 +1,46 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIPinMessageRequestPayload(UnverifiedPayloadBaseModel): + chat_id: UUID + sync_id: UUID + + @classmethod + def from_domain( + cls, + chat_id: UUID, + sync_id: UUID, + ) -> "BotXAPIPinMessageRequestPayload": + return cls(chat_id=chat_id, sync_id=sync_id) + + +class BotXAPIPinMessageResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class PinMessageMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIPinMessageRequestPayload, + ) -> None: + path = "/api/v3/botx/chats/pin_message" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPIPinMessageResponsePayload, response) diff --git a/botx/client/chats_api/remove_user.py b/botx/client/chats_api/remove_user.py new file mode 100644 index 00000000..8b0f2627 --- /dev/null +++ b/botx/client/chats_api/remove_user.py @@ -0,0 +1,46 @@ +from typing import List, Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIRemoveUserRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + user_huids: List[UUID] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + huids: List[UUID], + ) -> "BotXAPIRemoveUserRequestPayload": + return cls(group_chat_id=chat_id, user_huids=huids) + + +class BotXAPIRemoveUserResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class RemoveUserMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIRemoveUserRequestPayload, + ) -> None: + path = "/api/v3/botx/chats/remove_user" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPIRemoveUserResponsePayload, response) diff --git a/botx/client/chats_api/set_stealth.py b/botx/client/chats_api/set_stealth.py new file mode 100644 index 00000000..b0e0d012 --- /dev/null +++ b/botx/client/chats_api/set_stealth.py @@ -0,0 +1,53 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.missing import Missing +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPISetStealthRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + disable_web: Missing[bool] + burn_in: Missing[int] + expire_in: Missing[int] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + disable_web_client: Missing[bool], + ttl_after_read: Missing[int], + total_ttl: Missing[int], + ) -> "BotXAPISetStealthRequestPayload": + return cls( + group_chat_id=chat_id, + disable_web=disable_web_client, + burn_in=ttl_after_read, + expire_in=total_ttl, + ) + + +class BotXAPISetStealthResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class SetStealthMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute(self, payload: BotXAPISetStealthRequestPayload) -> None: + path = "/api/v3/botx/chats/stealth_set" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPISetStealthResponsePayload, response) diff --git a/botx/client/chats_api/unpin_message.py b/botx/client/chats_api/unpin_message.py new file mode 100644 index 00000000..d1260f02 --- /dev/null +++ b/botx/client/chats_api/unpin_message.py @@ -0,0 +1,44 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError, PermissionDeniedError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIUnpinMessageRequestPayload(UnverifiedPayloadBaseModel): + chat_id: UUID + + @classmethod + def from_domain( + cls, + chat_id: UUID, + ) -> "BotXAPIUnpinMessageRequestPayload": + return cls(chat_id=chat_id) + + +class BotXAPIUnpinMessageResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class UnpinMessageMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 403: response_exception_thrower(PermissionDeniedError), + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIUnpinMessageRequestPayload, + ) -> None: + path = "/api/v3/botx/chats/unpin_message" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPIUnpinMessageResponsePayload, response) diff --git a/docs/src/development/logging/__init__.py b/botx/client/events_api/__init__.py similarity index 100% rename from docs/src/development/logging/__init__.py rename to botx/client/events_api/__init__.py diff --git a/botx/client/events_api/edit_event.py b/botx/client/events_api/edit_event.py new file mode 100644 index 00000000..8bfeb855 --- /dev/null +++ b/botx/client/events_api/edit_event.py @@ -0,0 +1,98 @@ +from typing import Any, Dict, List, Literal, Union +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.missing import Missing, MissingOptional, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.attachments import ( + BotXAPIAttachment, + IncomingFileAttachment, + OutgoingAttachment, +) +from botx.models.message.markup import ( + BotXAPIMarkup, + BubbleMarkup, + KeyboardMarkup, + api_markup_from_domain, +) +from botx.models.message.mentions import BotXAPIMention, find_and_replace_embed_mentions + + +class BotXAPIEditEventOpts(UnverifiedPayloadBaseModel): + buttons_auto_adjust: Missing[bool] + + +class BotXAPIEditEvent(UnverifiedPayloadBaseModel): + body: Missing[str] + metadata: Missing[Dict[str, Any]] + opts: Missing[BotXAPIEditEventOpts] + bubble: Missing[BotXAPIMarkup] + keyboard: Missing[BotXAPIMarkup] + mentions: Missing[List[BotXAPIMention]] + + +class BotXAPIEditEventRequestPayload(UnverifiedPayloadBaseModel): + sync_id: UUID + payload: BotXAPIEditEvent + file: Missing[BotXAPIAttachment] + + @classmethod + def from_domain( + cls, + sync_id: UUID, + body: Missing[str], + metadata: Missing[Dict[str, Any]], + bubbles: Missing[BubbleMarkup], + keyboard: Missing[KeyboardMarkup], + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment, None]], + markup_auto_adjust: Missing[bool], + ) -> "BotXAPIEditEventRequestPayload": + api_file: MissingOptional[BotXAPIAttachment] = Undefined + if file: + assert not file.is_async_file, "async_files not supported" + api_file = BotXAPIAttachment.from_file_attachment(file) + elif file is None: + api_file = None + + mentions: Missing[List[BotXAPIMention]] = Undefined + if isinstance(body, str): + body, mentions = find_and_replace_embed_mentions(body) + + return cls( + sync_id=sync_id, + payload=BotXAPIEditEvent( + body=body, + # TODO: Metadata can be cleaned with `{}` + metadata=metadata, + opts=BotXAPIEditEventOpts( + buttons_auto_adjust=markup_auto_adjust, + ), + bubble=api_markup_from_domain(bubbles) if bubbles else bubbles, + keyboard=api_markup_from_domain(keyboard) if keyboard else keyboard, + mentions=mentions, + ), + file=api_file, + ) + + +class BotXAPIEditEventResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class EditEventMethod(AuthorizedBotXMethod): + async def execute( + self, + payload: BotXAPIEditEventRequestPayload, + ) -> None: + path = "/api/v3/botx/events/edit_event" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model( + BotXAPIEditEventResponsePayload, + response, + ) diff --git a/botx/client/events_api/message_status_event.py b/botx/client/events_api/message_status_event.py new file mode 100644 index 00000000..89a3071c --- /dev/null +++ b/botx/client/events_api/message_status_event.py @@ -0,0 +1,79 @@ +from dataclasses import dataclass +from datetime import datetime +from typing import List, Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.event import EventNotFoundError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.message.message_status import MessageStatus + + +class BotXAPIMessageStatusRequestPayload(UnverifiedPayloadBaseModel): + sync_id: UUID + + @classmethod + def from_domain(cls, sync_id: UUID) -> "BotXAPIMessageStatusRequestPayload": + return cls(sync_id=sync_id) + + +@dataclass +class BotXAPIMessageStatusReadUser: + user_huid: UUID + read_at: datetime + + +@dataclass +class BotXAPIMessageStatusReceivedUser: + user_huid: UUID + received_at: datetime + + +class BotXAPIMessageStatusResult(VerifiedPayloadBaseModel): + group_chat_id: UUID + sent_to: List[UUID] + read_by: List[BotXAPIMessageStatusReadUser] + received_by: List[BotXAPIMessageStatusReceivedUser] + + +class BotXAPIMessageStatusResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIMessageStatusResult + + def to_domain(self) -> MessageStatus: + + return MessageStatus( + group_chat_id=self.result.group_chat_id, + sent_to=self.result.sent_to, + read_by={ + reader.user_huid: reader.read_at for reader in self.result.read_by + }, + received_by={ + receiver.user_huid: receiver.received_at + for receiver in self.result.received_by + }, + ) + + +class MessageStatusMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(EventNotFoundError), + } + + async def execute( + self, + payload: BotXAPIMessageStatusRequestPayload, + ) -> "BotXAPIMessageStatusResponsePayload": + path = f"/api/v3/botx/events/{payload.sync_id}/status" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + ) + + return self._verify_and_extract_api_model( + BotXAPIMessageStatusResponsePayload, + response, + ) diff --git a/botx/client/events_api/reply_event.py b/botx/client/events_api/reply_event.py new file mode 100644 index 00000000..3275db25 --- /dev/null +++ b/botx/client/events_api/reply_event.py @@ -0,0 +1,118 @@ +from typing import Any, Dict, List, Literal, Union +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.missing import Missing, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.attachments import ( + BotXAPIAttachment, + IncomingFileAttachment, + OutgoingAttachment, +) +from botx.models.message.markup import ( + BotXAPIMarkup, + BubbleMarkup, + KeyboardMarkup, + api_markup_from_domain, +) +from botx.models.message.mentions import BotXAPIMention, find_and_replace_embed_mentions + + +class BotXAPIReplyEventMessageOpts(UnverifiedPayloadBaseModel): + silent_response: Missing[bool] + buttons_auto_adjust: Missing[bool] + + +class BotXAPIReplyEvent(UnverifiedPayloadBaseModel): + status: Literal["ok"] + body: str + metadata: Missing[Dict[str, Any]] + opts: Missing[BotXAPIReplyEventMessageOpts] + bubble: Missing[BotXAPIMarkup] + keyboard: Missing[BotXAPIMarkup] + mentions: Missing[List[BotXAPIMention]] + + +class BotXAPIReplyEventNestedOpts(UnverifiedPayloadBaseModel): + send: Missing[bool] + force_dnd: Missing[bool] + + +class BotXAPIReplyEventOpts(UnverifiedPayloadBaseModel): + raw_mentions: Literal[True] + stealth_mode: Missing[bool] + notification_opts: Missing[BotXAPIReplyEventNestedOpts] + + +class BotXAPIReplyEventRequestPayload(UnverifiedPayloadBaseModel): + source_sync_id: UUID + reply: BotXAPIReplyEvent + file: Missing[BotXAPIAttachment] + opts: BotXAPIReplyEventOpts + + @classmethod + def from_domain( + cls, + sync_id: UUID, + body: str, + metadata: Missing[Dict[str, Any]], + bubbles: Missing[BubbleMarkup], + keyboard: Missing[KeyboardMarkup], + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]], + silent_response: Missing[bool], + markup_auto_adjust: Missing[bool], + stealth_mode: Missing[bool], + send_push: Missing[bool], + ignore_mute: Missing[bool], + ) -> "BotXAPIReplyEventRequestPayload": + api_file: Missing[BotXAPIAttachment] = Undefined + if file: + assert not file.is_async_file, "async_files not supported" + api_file = BotXAPIAttachment.from_file_attachment(file) + + body, mentions = find_and_replace_embed_mentions(body) + + return cls( + source_sync_id=sync_id, + reply=BotXAPIReplyEvent( + status="ok", + body=body, + metadata=metadata, + opts=BotXAPIReplyEventMessageOpts( + buttons_auto_adjust=markup_auto_adjust, + silent_response=silent_response, + ), + bubble=api_markup_from_domain(bubbles) if bubbles else bubbles, + keyboard=api_markup_from_domain(keyboard) if keyboard else keyboard, + mentions=mentions or Undefined, + ), + file=api_file, + opts=BotXAPIReplyEventOpts( + raw_mentions=True, + stealth_mode=stealth_mode, + notification_opts=BotXAPIReplyEventNestedOpts( + send=send_push, + force_dnd=ignore_mute, + ), + ), + ) + + +class BotXAPIReplyEventResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class ReplyEventMethod(AuthorizedBotXMethod): + async def execute(self, payload: BotXAPIReplyEventRequestPayload) -> None: + path = "/api/v3/botx/events/reply_event" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model( + BotXAPIReplyEventResponsePayload, + response, + ) diff --git a/botx/client/events_api/stop_typing_event.py b/botx/client/events_api/stop_typing_event.py new file mode 100644 index 00000000..e9d27298 --- /dev/null +++ b/botx/client/events_api/stop_typing_event.py @@ -0,0 +1,33 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIStopTypingEventRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + + @classmethod + def from_domain(cls, chat_id: UUID) -> "BotXAPIStopTypingEventRequestPayload": + return cls(group_chat_id=chat_id) + + +class BotXAPIStopTypingEventResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class StopTypingEventMethod(AuthorizedBotXMethod): + async def execute(self, payload: BotXAPIStopTypingEventRequestPayload) -> None: + path = "/api/v3/botx/events/stop_typing" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model( + BotXAPIStopTypingEventResponsePayload, + response, + ) diff --git a/botx/client/events_api/typing_event.py b/botx/client/events_api/typing_event.py new file mode 100644 index 00000000..e7f76e5d --- /dev/null +++ b/botx/client/events_api/typing_event.py @@ -0,0 +1,30 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPITypingEventRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + + @classmethod + def from_domain(cls, chat_id: UUID) -> "BotXAPITypingEventRequestPayload": + return cls(group_chat_id=chat_id) + + +class BotXAPITypingEventResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class TypingEventMethod(AuthorizedBotXMethod): + async def execute(self, payload: BotXAPITypingEventRequestPayload) -> None: + path = "/api/v3/botx/events/typing" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model(BotXAPITypingEventResponsePayload, response) diff --git a/docs/src/development/sending_data/__init__.py b/botx/client/exceptions/__init__.py similarity index 100% rename from docs/src/development/sending_data/__init__.py rename to botx/client/exceptions/__init__.py diff --git a/botx/client/exceptions/base.py b/botx/client/exceptions/base.py new file mode 100644 index 00000000..68ddf2bc --- /dev/null +++ b/botx/client/exceptions/base.py @@ -0,0 +1,49 @@ +from typing import Optional + +import httpx + +from botx.models.method_callbacks import BotAPIMethodFailedCallback + + +class BaseClientError(Exception): + def __init__(self, message: str) -> None: + self.message = message + super().__init__(message) + + @classmethod + def from_response( + cls, + response: httpx.Response, + comment: Optional[str] = None, + ) -> "BaseClientError": + method = response.request.method + url = response.request.url + status_code = response.status_code + content = response.content + + message = ( + f"{method} {url}\n" # noqa: WPS221 (Strange error on CI) + f"failed with code {status_code} and payload:\n" + f"{content!r}" + ) + + if comment is not None: + message = f"{message}\n\nComment: {comment}" + + return cls(message) + + @classmethod + def from_callback( + cls, + callback: BotAPIMethodFailedCallback, + comment: Optional[str] = None, + ) -> "BaseClientError": + message = ( + f"BotX method call with sync_id `{callback.sync_id!s}` " + f"failed with: {callback}" + ) + + if comment is not None: + message = f"{message}\n\nComment: {comment}" + + return cls(message) diff --git a/botx/client/exceptions/callbacks.py b/botx/client/exceptions/callbacks.py new file mode 100644 index 00000000..5a28325d --- /dev/null +++ b/botx/client/exceptions/callbacks.py @@ -0,0 +1,19 @@ +from uuid import UUID + +from botx.client.exceptions.base import BaseClientError +from botx.models.method_callbacks import BotAPIMethodFailedCallback + + +class BotXMethodFailedCallbackReceivedError(BaseClientError): + """Callback with error received.""" + + def __init__(self, callback: BotAPIMethodFailedCallback) -> None: + exc = BaseClientError.from_callback(callback) + self.args = exc.args + + +class CallbackNotReceivedError(Exception): + def __init__(self, sync_id: UUID) -> None: + self.sync_id = sync_id + self.message = f"Callback for sync_id `{sync_id}` hasn't been received" + super().__init__(self.message) diff --git a/botx/client/exceptions/chats.py b/botx/client/exceptions/chats.py new file mode 100644 index 00000000..cd821bd5 --- /dev/null +++ b/botx/client/exceptions/chats.py @@ -0,0 +1,17 @@ +from botx.client.exceptions.base import BaseClientError + + +class CantUpdatePersonalChatError(BaseClientError): + """Can't edit a personal chat.""" + + +class InvalidUsersListError(BaseClientError): + """Users list isn't correct.""" + + +class ChatCreationProhibitedError(BaseClientError): + """Bot doesn't have permissions to create chat.""" + + +class ChatCreationError(BaseClientError): + """Error while chat creation.""" diff --git a/botx/client/exceptions/common.py b/botx/client/exceptions/common.py new file mode 100644 index 00000000..02496b14 --- /dev/null +++ b/botx/client/exceptions/common.py @@ -0,0 +1,17 @@ +from botx.client.exceptions.base import BaseClientError + + +class InvalidBotAccountError(BaseClientError): + """Can't get token with given bot account.""" + + +class RateLimitReachedError(BaseClientError): + """Too many method requests.""" + + +class PermissionDeniedError(BaseClientError): + """Bot can't perform this action.""" + + +class ChatNotFoundError(BaseClientError): + """Chat with specified group_chat_id not found.""" diff --git a/botx/client/exceptions/event.py b/botx/client/exceptions/event.py new file mode 100644 index 00000000..9ffd7d4b --- /dev/null +++ b/botx/client/exceptions/event.py @@ -0,0 +1,5 @@ +from botx.client.exceptions.base import BaseClientError + + +class EventNotFoundError(BaseClientError): + """Event not found.""" diff --git a/botx/client/exceptions/files.py b/botx/client/exceptions/files.py new file mode 100644 index 00000000..6002e7f5 --- /dev/null +++ b/botx/client/exceptions/files.py @@ -0,0 +1,9 @@ +from botx.client.exceptions.base import BaseClientError + + +class FileDeletedError(BaseClientError): + """File deleted.""" + + +class FileMetadataNotFound(BaseClientError): + """Can't find file metadata.""" diff --git a/botx/client/exceptions/http.py b/botx/client/exceptions/http.py new file mode 100644 index 00000000..e7bdbb41 --- /dev/null +++ b/botx/client/exceptions/http.py @@ -0,0 +1,19 @@ +import httpx + +from botx.client.exceptions.base import BaseClientError + + +class InvalidBotXResponseError(BaseClientError): + """Received invalid response.""" + + def __init__(self, response: httpx.Response) -> None: + exc = BaseClientError.from_response(response) + self.args = exc.args + + +class InvalidBotXStatusCodeError(InvalidBotXResponseError): + """Received invalid status code.""" + + +class InvalidBotXResponsePayloadError(InvalidBotXResponseError): + """Received invalid status code.""" diff --git a/botx/client/exceptions/notifications.py b/botx/client/exceptions/notifications.py new file mode 100644 index 00000000..6795327e --- /dev/null +++ b/botx/client/exceptions/notifications.py @@ -0,0 +1,13 @@ +from botx.client.exceptions.base import BaseClientError + + +class BotIsNotChatMemberError(BaseClientError): + """Bot is not in the list of chat members.""" + + +class FinalRecipientsListEmptyError(BaseClientError): + """Resulting event recipients list is empty.""" + + +class StealthModeDisabledError(BaseClientError): + """Requested stealth mode disabled in specified chat.""" diff --git a/botx/client/exceptions/users.py b/botx/client/exceptions/users.py new file mode 100644 index 00000000..3263bf15 --- /dev/null +++ b/botx/client/exceptions/users.py @@ -0,0 +1,5 @@ +from botx.client.exceptions.base import BaseClientError + + +class UserNotFoundError(BaseClientError): + """User not found.""" diff --git a/docs/src/development/tests/__init__.py b/botx/client/files_api/__init__.py similarity index 100% rename from docs/src/development/tests/__init__.py rename to botx/client/files_api/__init__.py diff --git a/botx/client/files_api/download_file.py b/botx/client/files_api/download_file.py new file mode 100644 index 00000000..8beaf4a4 --- /dev/null +++ b/botx/client/files_api/download_file.py @@ -0,0 +1,67 @@ +from typing import NoReturn +from uuid import UUID + +import httpx + +from botx.async_buffer import AsyncBufferWritable +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError +from botx.client.exceptions.files import FileDeletedError, FileMetadataNotFound +from botx.client.exceptions.http import InvalidBotXStatusCodeError +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPIDownloadFileRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + file_id: UUID + is_preview: bool + + @classmethod + def from_domain( + cls, + chat_id: UUID, + file_id: UUID, + ) -> "BotXAPIDownloadFileRequestPayload": + return cls( + group_chat_id=chat_id, + file_id=file_id, + is_preview=False, + ) + + +def not_found_error_handler(response: httpx.Response) -> NoReturn: + reason = response.json().get("reason") + + if reason == "file_metadata_not_found": + raise FileMetadataNotFound.from_response(response) + elif reason == "chat_not_found": + raise ChatNotFoundError.from_response(response) + + raise InvalidBotXStatusCodeError(response) + + +class DownloadFileMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 204: response_exception_thrower(FileDeletedError), + 404: not_found_error_handler, + } + + async def execute( + self, + payload: BotXAPIDownloadFileRequestPayload, + async_buffer: AsyncBufferWritable, + ) -> None: + path = "/api/v3/botx/files/download" + + async with self._botx_method_stream( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) as response: + # https://github.com/nedbat/coveragepy/issues/1223 + async for chunk in response.aiter_bytes(): # pragma: no cover + await async_buffer.write(chunk) + + await async_buffer.seek(0) diff --git a/botx/client/files_api/upload_file.py b/botx/client/files_api/upload_file.py new file mode 100644 index 00000000..2c8e7408 --- /dev/null +++ b/botx/client/files_api/upload_file.py @@ -0,0 +1,80 @@ +import tempfile +from typing import Literal +from uuid import UUID + +from botx.async_buffer import AsyncBufferReadable +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError +from botx.constants import CHUNK_SIZE +from botx.missing import Missing +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.async_files import APIAsyncFile, File, convert_async_file_to_domain + + +class BotXAPIUploadFileMeta(UnverifiedPayloadBaseModel): + duration: Missing[int] + caption: Missing[str] + + +class BotXAPIUploadFileRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + meta: str + + @classmethod + def from_domain( + cls, + chat_id: UUID, + duration: Missing[int], + caption: Missing[str], + ) -> "BotXAPIUploadFileRequestPayload": + return cls( + group_chat_id=chat_id, + meta=BotXAPIUploadFileMeta( + duration=duration, + caption=caption, + ).json(), + ) + + +class BotXAPIUploadFileResponsePayload(VerifiedPayloadBaseModel): + result: APIAsyncFile + status: Literal["ok"] + + def to_domain(self) -> File: + return convert_async_file_to_domain(self.result) + + +class UploadFileMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(ChatNotFoundError), + } + + async def execute( + self, + payload: BotXAPIUploadFileRequestPayload, + async_buffer: AsyncBufferReadable, + filename: str, + ) -> BotXAPIUploadFileResponsePayload: + path = "/api/v3/botx/files/upload" + + with tempfile.SpooledTemporaryFile(max_size=CHUNK_SIZE) as tmp_file: + chunk = await async_buffer.read(CHUNK_SIZE) + while chunk: + tmp_file.write(chunk) + chunk = await async_buffer.read(CHUNK_SIZE) + + tmp_file.seek(0) + + response = await self._botx_method_call( + "POST", + self._build_url(path), + data=payload.jsonable_dict(), + files={"content": (filename, tmp_file)}, + ) + + return self._verify_and_extract_api_model( + BotXAPIUploadFileResponsePayload, + response, + ) diff --git a/botx/client/get_token.py b/botx/client/get_token.py new file mode 100644 index 00000000..e353f2f2 --- /dev/null +++ b/botx/client/get_token.py @@ -0,0 +1,30 @@ +from uuid import UUID + +import httpx + +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.client.bots_api.get_token import BotXAPIGetTokenRequestPayload, GetTokenMethod + + +async def get_token( + bot_id: UUID, + httpx_client: httpx.AsyncClient, + bot_accounts_storage: BotAccountsStorage, +) -> str: # noqa: DAR101, DAR201 + """Request token for bot. + + Moved to separate file because used in `AuthorizedBotXMethod` and `Bot.get_token`. + """ + + method = GetTokenMethod( + bot_id, + httpx_client, + bot_accounts_storage, + ) + + signature = bot_accounts_storage.build_signature(bot_id) + payload = BotXAPIGetTokenRequestPayload.from_domain(signature) + + botx_api_token = await method.execute(payload) + + return botx_api_token.to_domain() diff --git a/docs/src/development/tests/tests0/__init__.py b/botx/client/notifications_api/__init__.py similarity index 100% rename from docs/src/development/tests/tests0/__init__.py rename to botx/client/notifications_api/__init__.py diff --git a/botx/client/notifications_api/direct_notification.py b/botx/client/notifications_api/direct_notification.py new file mode 100644 index 00000000..939eb32f --- /dev/null +++ b/botx/client/notifications_api/direct_notification.py @@ -0,0 +1,164 @@ +from typing import Any, Dict, List, Literal, Optional, Union +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import callback_exception_thrower +from botx.client.exceptions.common import ChatNotFoundError +from botx.client.exceptions.notifications import ( + BotIsNotChatMemberError, + FinalRecipientsListEmptyError, + StealthModeDisabledError, +) +from botx.constants import MAX_NOTIFICATION_BODY_LENGTH +from botx.missing import Missing, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.attachments import ( + BotXAPIAttachment, + IncomingFileAttachment, + OutgoingAttachment, +) +from botx.models.message.markup import ( + BotXAPIMarkup, + BubbleMarkup, + KeyboardMarkup, + api_markup_from_domain, +) +from botx.models.message.mentions import BotXAPIMention, find_and_replace_embed_mentions + + +class BotXAPIDirectNotificationMessageOpts(UnverifiedPayloadBaseModel): + silent_response: Missing[bool] + buttons_auto_adjust: Missing[bool] + + +class BotXAPIDirectNotificationNestedOpts(UnverifiedPayloadBaseModel): + send: Missing[bool] + force_dnd: Missing[bool] + + +class BotXAPIDirectNotificationOpts(UnverifiedPayloadBaseModel): + stealth_mode: Missing[bool] + notification_opts: Missing[BotXAPIDirectNotificationNestedOpts] + + +class BotXAPIDirectNotification(UnverifiedPayloadBaseModel): + status: Literal["ok"] + body: str + metadata: Missing[Dict[str, Any]] + opts: Missing[BotXAPIDirectNotificationMessageOpts] + bubble: Missing[BotXAPIMarkup] + keyboard: Missing[BotXAPIMarkup] + mentions: Missing[List[BotXAPIMention]] + + +class BotXAPIDirectNotificationRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + notification: BotXAPIDirectNotification + file: Missing[BotXAPIAttachment] + recipients: Missing[List[UUID]] + opts: Missing[BotXAPIDirectNotificationOpts] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + body: str, + metadata: Missing[Dict[str, Any]], + bubbles: Missing[BubbleMarkup], + keyboard: Missing[KeyboardMarkup], + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]], + recipients: Missing[List[UUID]], + silent_response: Missing[bool], + markup_auto_adjust: Missing[bool], + stealth_mode: Missing[bool], + send_push: Missing[bool], + ignore_mute: Missing[bool], + ) -> "BotXAPIDirectNotificationRequestPayload": + api_file: Missing[BotXAPIAttachment] = Undefined + if file: + assert not file.is_async_file, "async_files not supported" + api_file = BotXAPIAttachment.from_file_attachment(file) + + if len(body) > MAX_NOTIFICATION_BODY_LENGTH: + raise ValueError( + f"Message body length exceeds {MAX_NOTIFICATION_BODY_LENGTH} symbols", + ) + + body, mentions = find_and_replace_embed_mentions(body) + + return cls( + group_chat_id=chat_id, + notification=BotXAPIDirectNotification( + status="ok", + body=body, + metadata=metadata, + opts=BotXAPIDirectNotificationMessageOpts( + silent_response=silent_response, + buttons_auto_adjust=markup_auto_adjust, + ), + bubble=api_markup_from_domain(bubbles) if bubbles else bubbles, + keyboard=api_markup_from_domain(keyboard) if keyboard else keyboard, + mentions=mentions or Undefined, + ), + file=api_file, + recipients=recipients, + opts=BotXAPIDirectNotificationOpts( + stealth_mode=stealth_mode, + notification_opts=BotXAPIDirectNotificationNestedOpts( + send=send_push, + force_dnd=ignore_mute, + ), + ), + ) + + +class BotXAPISyncIdResult(VerifiedPayloadBaseModel): + sync_id: UUID + + +class BotXAPIDirectNotificationResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPISyncIdResult + + def to_domain(self) -> UUID: + return self.result.sync_id + + +class DirectNotificationMethod(AuthorizedBotXMethod): + error_callback_handlers = { + **AuthorizedBotXMethod.error_callback_handlers, + "chat_not_found": callback_exception_thrower(ChatNotFoundError), + "bot_is_not_a_chat_member": callback_exception_thrower( + BotIsNotChatMemberError, + ), + "event_recipients_list_is_empty": callback_exception_thrower( + FinalRecipientsListEmptyError, + ), + "stealth_mode_disabled": callback_exception_thrower(StealthModeDisabledError), + } + + async def execute( + self, + payload: BotXAPIDirectNotificationRequestPayload, + wait_callback: bool, + callback_timeout: Optional[int], + ) -> BotXAPIDirectNotificationResponsePayload: + path = "/api/v4/botx/notifications/direct" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + api_model = self._verify_and_extract_api_model( + BotXAPIDirectNotificationResponsePayload, + response, + ) + + await self._process_callback( + api_model.result.sync_id, + wait_callback, + callback_timeout, + ) + return api_model diff --git a/botx/client/notifications_api/internal_bot_notification.py b/botx/client/notifications_api/internal_bot_notification.py new file mode 100644 index 00000000..97722418 --- /dev/null +++ b/botx/client/notifications_api/internal_bot_notification.py @@ -0,0 +1,93 @@ +from typing import Any, Dict, List, Literal, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import ( + callback_exception_thrower, + response_exception_thrower, +) +from botx.client.exceptions.common import ChatNotFoundError, RateLimitReachedError +from botx.client.exceptions.notifications import ( + BotIsNotChatMemberError, + FinalRecipientsListEmptyError, +) +from botx.missing import Missing, MissingOptional +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIInternalBotNotificationRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + data: Dict[str, Any] + opts: Missing[Dict[str, Any]] + recipients: MissingOptional[List[UUID]] + + @classmethod + def from_domain( + cls, + chat_id: UUID, + data: Dict[str, Any], + opts: Missing[Dict[str, Any]], + recipients: MissingOptional[List[UUID]], + ) -> "BotXAPIInternalBotNotificationRequestPayload": + return cls( + group_chat_id=chat_id, + data=data, + opts=opts, + recipients=recipients, + ) + + +class BotXAPISyncIdResult(VerifiedPayloadBaseModel): + sync_id: UUID + + +class BotXAPIInternalBotNotificationResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPISyncIdResult + + def to_domain(self) -> UUID: + return self.result.sync_id + + +class InternalBotNotificationMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 429: response_exception_thrower(RateLimitReachedError), + } + + error_callback_handlers = { + **AuthorizedBotXMethod.error_callback_handlers, + "chat_not_found": callback_exception_thrower(ChatNotFoundError), + "bot_is_not_a_chat_member": callback_exception_thrower( + BotIsNotChatMemberError, + ), + "event_recipients_list_is_empty": callback_exception_thrower( + FinalRecipientsListEmptyError, + ), + } + + async def execute( + self, + payload: BotXAPIInternalBotNotificationRequestPayload, + wait_callback: bool, + callback_timeout: Optional[int], + ) -> BotXAPIInternalBotNotificationResponsePayload: + path = "/api/v4/botx/notifications/internal" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + api_model = self._verify_and_extract_api_model( + BotXAPIInternalBotNotificationResponsePayload, + response, + ) + + await self._process_callback( + api_model.result.sync_id, + wait_callback, + callback_timeout, + ) + + return api_model diff --git a/docs/src/index/__init__.py b/botx/client/smartapps_api/__init__.py similarity index 100% rename from docs/src/index/__init__.py rename to botx/client/smartapps_api/__init__.py diff --git a/botx/client/smartapps_api/smartapp_event.py b/botx/client/smartapps_api/smartapp_event.py new file mode 100644 index 00000000..e9f39b37 --- /dev/null +++ b/botx/client/smartapps_api/smartapp_event.py @@ -0,0 +1,70 @@ +from typing import Any, Dict, List, Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.constants import SMARTAPP_API_VERSION +from botx.missing import Missing, MissingOptional, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.async_files import APIAsyncFile, File, convert_async_file_from_domain + + +class BotXAPISmartAppEventRequestPayload(UnverifiedPayloadBaseModel): + ref: MissingOptional[UUID] + smartapp_id: UUID + group_chat_id: UUID + data: Dict[str, Any] + opts: Missing[Dict[str, Any]] + smartapp_api_version: int + async_files: Missing[List[APIAsyncFile]] + + @classmethod + def from_domain( + cls, + ref: MissingOptional[UUID], + smartapp_id: UUID, + chat_id: UUID, + data: Dict[str, Any], + opts: Missing[Dict[str, Any]], + files: Missing[List[File]], + ) -> "BotXAPISmartAppEventRequestPayload": + api_async_files: Missing[List[APIAsyncFile]] = Undefined + if files: + api_async_files = [convert_async_file_from_domain(file) for file in files] + + return cls( + ref=ref, + smartapp_id=smartapp_id, + group_chat_id=chat_id, + data=data, + opts=opts, + smartapp_api_version=SMARTAPP_API_VERSION, + async_files=api_async_files, + ) + + +class BotXAPISmartAppEventResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class SmartAppEventMethod(AuthorizedBotXMethod): + async def execute( + self, + payload: BotXAPISmartAppEventRequestPayload, + ) -> None: + path = "/api/v3/botx/smartapps/event" + + # TODO: Remove opts + # UnverifiedPayloadBaseModel.jsonable_dict remove empty dicts + json = payload.jsonable_dict() + json["opts"] = json.get("opts", {}) + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=json, + ) + + self._verify_and_extract_api_model( + BotXAPISmartAppEventResponsePayload, + response, + ) diff --git a/botx/client/smartapps_api/smartapp_notification.py b/botx/client/smartapps_api/smartapp_notification.py new file mode 100644 index 00000000..cf69f9b1 --- /dev/null +++ b/botx/client/smartapps_api/smartapp_notification.py @@ -0,0 +1,56 @@ +from typing import Any, Dict, Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.constants import SMARTAPP_API_VERSION +from botx.missing import Missing +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPISmartAppNotificationRequestPayload(UnverifiedPayloadBaseModel): + group_chat_id: UUID + smartapp_counter: int + opts: Missing[Dict[str, Any]] + smartapp_api_version: int + + @classmethod + def from_domain( + cls, + chat_id: UUID, + smartapp_counter: int, + opts: Missing[Dict[str, Any]], + ) -> "BotXAPISmartAppNotificationRequestPayload": + return cls( + group_chat_id=chat_id, + smartapp_counter=smartapp_counter, + opts=opts, + smartapp_api_version=SMARTAPP_API_VERSION, + ) + + +class BotXAPISmartAppNotificationResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class SmartAppNotificationMethod(AuthorizedBotXMethod): + async def execute( + self, + payload: BotXAPISmartAppNotificationRequestPayload, + ) -> None: + path = "/api/v3/botx/smartapps/notification" + + # TODO: Remove opts + # UnverifiedPayloadBaseModel.jsonable_dict remove empty dicts + json = payload.jsonable_dict() + json["opts"] = json.get("opts", {}) + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=json, + ) + + self._verify_and_extract_api_model( + BotXAPISmartAppNotificationResponsePayload, + response, + ) diff --git a/examples/fsm/bot/__init__.py b/botx/client/stickers_api/__init__.py similarity index 100% rename from examples/fsm/bot/__init__.py rename to botx/client/stickers_api/__init__.py diff --git a/botx/client/stickers_api/add_sticker.py b/botx/client/stickers_api/add_sticker.py new file mode 100644 index 00000000..6a2c6f26 --- /dev/null +++ b/botx/client/stickers_api/add_sticker.py @@ -0,0 +1,97 @@ +from typing import Literal, NoReturn +from uuid import UUID + +import httpx + +from botx.async_buffer import AsyncBufferReadable +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.exceptions.http import InvalidBotXStatusCodeError +from botx.client.stickers_api.exceptions import ( + InvalidEmojiError, + InvalidImageError, + StickerPackOrStickerNotFoundError, +) +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.attachments import encode_rfc2397 +from botx.models.stickers import Sticker + + +class BotXAPIAddStickerRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + emoji: str + image: str + + @classmethod + async def from_domain( + cls, + sticker_pack_id: UUID, + emoji: str, + async_buffer: AsyncBufferReadable, + ) -> "BotXAPIAddStickerRequestPayload": + mimetype = "image/png" + + content = await async_buffer.read() + b64_content = encode_rfc2397(content, mimetype) + + return cls(sticker_pack_id=sticker_pack_id, emoji=emoji, image=b64_content) + + +class BotXAPIAddStickerResult(VerifiedPayloadBaseModel): + id: UUID + emoji: str + link: str + + +class BotXAPIAddStickerResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIAddStickerResult + + def to_domain(self) -> Sticker: + return Sticker( + id=self.result.id, + emoji=self.result.emoji, + image_link=self.result.link, + ) + + +def bad_request_error_handler(response: httpx.Response) -> NoReturn: # noqa: WPS238 + reason = response.json().get("reason") + + if reason == "pack_not_found": + raise StickerPackOrStickerNotFoundError.from_response(response) + + error_data = response.json().get("error_data") + + if error_data.get("emoji") == "invalid": + raise InvalidEmojiError.from_response(response) + elif error_data.get("image") == "invalid": + raise InvalidImageError.from_response(response) + + raise InvalidBotXStatusCodeError(response) + + +class AddStickerMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 400: bad_request_error_handler, + } + + async def execute( + self, + payload: BotXAPIAddStickerRequestPayload, + ) -> BotXAPIAddStickerResponsePayload: + jsonable_dict = payload.jsonable_dict() + sticker_pack_id = jsonable_dict.pop("sticker_pack_id") + + path = f"/api/v3/botx/stickers/packs/{sticker_pack_id}/stickers" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=jsonable_dict, + ) + + return self._verify_and_extract_api_model( + BotXAPIAddStickerResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/create_sticker_pack.py b/botx/client/stickers_api/create_sticker_pack.py new file mode 100644 index 00000000..b0e04b33 --- /dev/null +++ b/botx/client/stickers_api/create_sticker_pack.py @@ -0,0 +1,56 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.stickers import StickerPack + + +class BotXAPICreateStickerPackRequestPayload(UnverifiedPayloadBaseModel): + name: str + + @classmethod + def from_domain(cls, name: str) -> "BotXAPICreateStickerPackRequestPayload": + return cls(name=name) + + +class BotXAPICreateStickerPackResult(VerifiedPayloadBaseModel): + id: UUID + name: str + public: bool + + +class BotXAPICreateStickerPackResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPICreateStickerPackResult + + def to_domain(self) -> StickerPack: + return StickerPack( + id=self.result.id, + name=self.result.name, + is_public=self.result.public, + stickers=[], + ) + + +class CreateStickerPackMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + } + + async def execute( + self, + payload: BotXAPICreateStickerPackRequestPayload, + ) -> BotXAPICreateStickerPackResponsePayload: + path = "/api/v3/botx/stickers/packs" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPICreateStickerPackResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/delete_sticker.py b/botx/client/stickers_api/delete_sticker.py new file mode 100644 index 00000000..3bcefa4f --- /dev/null +++ b/botx/client/stickers_api/delete_sticker.py @@ -0,0 +1,50 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.stickers_api.exceptions import StickerPackOrStickerNotFoundError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIDeleteStickerRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + sticker_id: UUID + + @classmethod + async def from_domain( + cls, + sticker_pack_id: UUID, + sticker_id: UUID, + ) -> "BotXAPIDeleteStickerRequestPayload": + return cls(sticker_pack_id=sticker_pack_id, sticker_id=sticker_id) + + +class BotXAPIDeleteStickerResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class DeleteStickerMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(StickerPackOrStickerNotFoundError), + } + + async def execute( + self, + payload: BotXAPIDeleteStickerRequestPayload, + ) -> None: + path = ( + f"/api/v3/botx/stickers/packs/{payload.sticker_pack_id}" + f"/stickers/{payload.sticker_id}" + ) + + response = await self._botx_method_call( + "DELETE", + self._build_url(path), + ) + + self._verify_and_extract_api_model( + BotXAPIDeleteStickerResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/delete_sticker_pack.py b/botx/client/stickers_api/delete_sticker_pack.py new file mode 100644 index 00000000..1bcef0ed --- /dev/null +++ b/botx/client/stickers_api/delete_sticker_pack.py @@ -0,0 +1,45 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.stickers_api.exceptions import StickerPackOrStickerNotFoundError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIDeleteStickerPackRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + + @classmethod + def from_domain( + cls, + sticker_pack_id: UUID, + ) -> "BotXAPIDeleteStickerPackRequestPayload": + return cls(sticker_pack_id=sticker_pack_id) + + +class BotXAPIDeleteStickerPackResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class DeleteStickerPackMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(StickerPackOrStickerNotFoundError), + } + + async def execute( + self, + payload: BotXAPIDeleteStickerPackRequestPayload, + ) -> BotXAPIDeleteStickerPackResponsePayload: + path = f"/api/v3/botx/stickers/packs/{payload.sticker_pack_id}" + + response = await self._botx_method_call( + "DELETE", + self._build_url(path), + ) + + return self._verify_and_extract_api_model( + BotXAPIDeleteStickerPackResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/edit_sticker_pack.py b/botx/client/stickers_api/edit_sticker_pack.py new file mode 100644 index 00000000..191e51f6 --- /dev/null +++ b/botx/client/stickers_api/edit_sticker_pack.py @@ -0,0 +1,57 @@ +from typing import List, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.stickers_api.exceptions import StickerPackOrStickerNotFoundError +from botx.client.stickers_api.sticker_pack import BotXAPIGetStickerPackResponsePayload +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPIEditStickerPackRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + name: str + preview: UUID + stickers_order: Optional[List[UUID]] + + @classmethod + def from_domain( + cls, + sticker_pack_id: UUID, + name: str, + preview: UUID, + stickers_order: Optional[List[UUID]], + ) -> "BotXAPIEditStickerPackRequestPayload": + return cls( + sticker_pack_id=sticker_pack_id, + name=name, + preview=preview, + stickers_order=stickers_order, + ) + + +class EditStickerPackMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(StickerPackOrStickerNotFoundError), + } + + async def execute( + self, + payload: BotXAPIEditStickerPackRequestPayload, + ) -> BotXAPIGetStickerPackResponsePayload: + jsonable_dict = payload.jsonable_dict() + sticker_pack_id = jsonable_dict.pop("sticker_pack_id") + + path = f"/api/v3/botx/stickers/packs/{sticker_pack_id}" + + response = await self._botx_method_call( + "PUT", + self._build_url(path), + json=jsonable_dict, + ) + + return self._verify_and_extract_api_model( + BotXAPIGetStickerPackResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/exceptions.py b/botx/client/stickers_api/exceptions.py new file mode 100644 index 00000000..4e6af776 --- /dev/null +++ b/botx/client/stickers_api/exceptions.py @@ -0,0 +1,13 @@ +from botx.client.exceptions.base import BaseClientError + + +class StickerPackOrStickerNotFoundError(BaseClientError): + """Sticker pack or sticker with specified id not found.""" + + +class InvalidEmojiError(BaseClientError): + """Bad emoji.""" + + +class InvalidImageError(BaseClientError): + """Bad image.""" diff --git a/botx/client/stickers_api/get_sticker.py b/botx/client/stickers_api/get_sticker.py new file mode 100644 index 00000000..81a37c08 --- /dev/null +++ b/botx/client/stickers_api/get_sticker.py @@ -0,0 +1,66 @@ +from typing import Literal +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.stickers_api.exceptions import StickerPackOrStickerNotFoundError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.stickers import Sticker + + +class BotXAPIGetStickerRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + sticker_id: UUID + + @classmethod + def from_domain( + cls, + sticker_pack_id: UUID, + sticker_id: UUID, + ) -> "BotXAPIGetStickerRequestPayload": + return cls(sticker_pack_id=sticker_pack_id, sticker_id=sticker_id) + + +class BotXAPIGetStickerResult(VerifiedPayloadBaseModel): + id: UUID + emoji: str + link: str + + +class BotXAPIGetStickerResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIGetStickerResult + + def to_domain(self) -> Sticker: + return Sticker( + id=self.result.id, + emoji=self.result.emoji, + image_link=self.result.link, + ) + + +class GetStickerMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(StickerPackOrStickerNotFoundError), + } + + async def execute( + self, + payload: BotXAPIGetStickerRequestPayload, + ) -> BotXAPIGetStickerResponsePayload: + jsonable_dict = payload.jsonable_dict() + path = ( + f"/api/v3/botx/stickers/packs/{jsonable_dict['sticker_pack_id']}/" + f"stickers/{jsonable_dict['sticker_id']}" + ) + + response = await self._botx_method_call( + "GET", + self._build_url(path), + ) + + return self._verify_and_extract_api_model( + BotXAPIGetStickerResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/get_sticker_pack.py b/botx/client/stickers_api/get_sticker_pack.py new file mode 100644 index 00000000..443df163 --- /dev/null +++ b/botx/client/stickers_api/get_sticker_pack.py @@ -0,0 +1,42 @@ +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.stickers_api.exceptions import StickerPackOrStickerNotFoundError +from botx.client.stickers_api.sticker_pack import BotXAPIGetStickerPackResponsePayload +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPIGetStickerPackRequestPayload(UnverifiedPayloadBaseModel): + sticker_pack_id: UUID + + @classmethod + def from_domain( + cls, + sticker_pack_id: UUID, + ) -> "BotXAPIGetStickerPackRequestPayload": + return cls(sticker_pack_id=sticker_pack_id) + + +class GetStickerPackMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(StickerPackOrStickerNotFoundError), + } + + async def execute( + self, + payload: BotXAPIGetStickerPackRequestPayload, + ) -> BotXAPIGetStickerPackResponsePayload: + jsonable_dict = payload.jsonable_dict() + path = f"/api/v3/botx/stickers/packs/{jsonable_dict['sticker_pack_id']}" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + ) + + return self._verify_and_extract_api_model( + BotXAPIGetStickerPackResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/get_sticker_packs.py b/botx/client/stickers_api/get_sticker_packs.py new file mode 100644 index 00000000..6800d429 --- /dev/null +++ b/botx/client/stickers_api/get_sticker_packs.py @@ -0,0 +1,77 @@ +from typing import List, Literal, Optional +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.stickers import StickerPackFromList, StickerPackPage + + +class BotXAPIGetStickerPacksRequestPayload(UnverifiedPayloadBaseModel): + user_huid: UUID + limit: int + after: Optional[str] + + @classmethod + def from_domain( + cls, + huid: UUID, + limit: int, + after: Optional[str], + ) -> "BotXAPIGetStickerPacksRequestPayload": + return cls(user_huid=huid, limit=limit, after=after) + + +class BotXAPIGetPaginationResult(VerifiedPayloadBaseModel): + after: Optional[str] + + +class BotXAPIGetStickerPackResult(VerifiedPayloadBaseModel): + id: UUID + name: str + public: bool + stickers_count: int + stickers_order: Optional[List[UUID]] + + +class BotXAPIGetStickerPacksResult(VerifiedPayloadBaseModel): + packs: List[BotXAPIGetStickerPackResult] + pagination: BotXAPIGetPaginationResult + + +class BotXAPIGetStickerPacksResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIGetStickerPacksResult + + def to_domain(self) -> StickerPackPage: + return StickerPackPage( + sticker_packs=[ + StickerPackFromList( + id=sticker_pack.id, + name=sticker_pack.name, + is_public=sticker_pack.public, + stickers_count=sticker_pack.stickers_count, + sticker_ids=sticker_pack.stickers_order, + ) + for sticker_pack in self.result.packs + ], + after=self.result.pagination.after, + ) + + +class GetStickerPacksMethod(AuthorizedBotXMethod): + async def execute( + self, + payload: BotXAPIGetStickerPacksRequestPayload, + ) -> BotXAPIGetStickerPacksResponsePayload: + path = "/api/v3/botx/stickers/packs" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPIGetStickerPacksResponsePayload, + response, + ) diff --git a/botx/client/stickers_api/sticker_pack.py b/botx/client/stickers_api/sticker_pack.py new file mode 100644 index 00000000..a03b3e45 --- /dev/null +++ b/botx/client/stickers_api/sticker_pack.py @@ -0,0 +1,42 @@ +from typing import List, Literal, Optional +from uuid import UUID + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.stickers import Sticker, StickerPack + + +class BotXAPIGetStickerResult(VerifiedPayloadBaseModel): + id: UUID + emoji: str + link: str + + +class BotXAPIGetStickerPackResult(VerifiedPayloadBaseModel): + id: UUID + name: str + public: bool + stickers_order: Optional[List[UUID]] + stickers: List[BotXAPIGetStickerResult] + + +class BotXAPIGetStickerPackResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPIGetStickerPackResult + + def to_domain(self) -> StickerPack: + if self.result.stickers_order: + self.result.stickers.sort( + key=lambda pack: self.result.stickers_order.index( # type:ignore + pack.id, + ), + ) + + return StickerPack( + id=self.result.id, + name=self.result.name, + is_public=self.result.public, + stickers=[ + Sticker(id=sticker.id, emoji=sticker.emoji, image_link=sticker.link) + for sticker in self.result.stickers + ], + ) diff --git a/tests/fixtures/__init__.py b/botx/client/users_api/__init__.py similarity index 100% rename from tests/fixtures/__init__.py rename to botx/client/users_api/__init__.py diff --git a/botx/client/users_api/search_user_by_email.py b/botx/client/users_api/search_user_by_email.py new file mode 100644 index 00000000..0a820e79 --- /dev/null +++ b/botx/client/users_api/search_user_by_email.py @@ -0,0 +1,37 @@ +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.users import UserNotFoundError +from botx.client.users_api.user_from_search import BotXAPISearchUserResponsePayload +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPISearchUserByEmailRequestPayload(UnverifiedPayloadBaseModel): + email: str + + @classmethod + def from_domain(cls, email: str) -> "BotXAPISearchUserByEmailRequestPayload": + return cls(email=email) + + +class SearchUserByEmailMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(UserNotFoundError), + } + + async def execute( + self, + payload: BotXAPISearchUserByEmailRequestPayload, + ) -> BotXAPISearchUserResponsePayload: + path = "/api/v3/botx/users/by_email" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPISearchUserResponsePayload, + response, + ) diff --git a/botx/client/users_api/search_user_by_huid.py b/botx/client/users_api/search_user_by_huid.py new file mode 100644 index 00000000..5e806fbe --- /dev/null +++ b/botx/client/users_api/search_user_by_huid.py @@ -0,0 +1,39 @@ +from uuid import UUID + +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.users import UserNotFoundError +from botx.client.users_api.user_from_search import BotXAPISearchUserResponsePayload +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPISearchUserByHUIDRequestPayload(UnverifiedPayloadBaseModel): + user_huid: UUID + + @classmethod + def from_domain(cls, huid: UUID) -> "BotXAPISearchUserByHUIDRequestPayload": + return cls(user_huid=huid) + + +class SearchUserByHUIDMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(UserNotFoundError), + } + + async def execute( + self, + payload: BotXAPISearchUserByHUIDRequestPayload, + ) -> BotXAPISearchUserResponsePayload: + path = "/api/v3/botx/users/by_huid" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPISearchUserResponsePayload, + response, + ) diff --git a/botx/client/users_api/search_user_by_login.py b/botx/client/users_api/search_user_by_login.py new file mode 100644 index 00000000..34175e64 --- /dev/null +++ b/botx/client/users_api/search_user_by_login.py @@ -0,0 +1,42 @@ +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from botx.client.botx_method import response_exception_thrower +from botx.client.exceptions.users import UserNotFoundError +from botx.client.users_api.user_from_search import BotXAPISearchUserResponsePayload +from botx.models.api_base import UnverifiedPayloadBaseModel + + +class BotXAPISearchUserByLoginRequestPayload(UnverifiedPayloadBaseModel): + ad_login: str + ad_domain: str + + @classmethod + def from_domain( + cls, + ad_login: str, + ad_domain: str, + ) -> "BotXAPISearchUserByLoginRequestPayload": + return cls(ad_login=ad_login, ad_domain=ad_domain) + + +class SearchUserByLoginMethod(AuthorizedBotXMethod): + status_handlers = { + **AuthorizedBotXMethod.status_handlers, + 404: response_exception_thrower(UserNotFoundError), + } + + async def execute( + self, + payload: BotXAPISearchUserByLoginRequestPayload, + ) -> BotXAPISearchUserResponsePayload: + path = "/api/v3/botx/users/by_login" + + response = await self._botx_method_call( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPISearchUserResponsePayload, + response, + ) diff --git a/botx/client/users_api/user_from_search.py b/botx/client/users_api/user_from_search.py new file mode 100644 index 00000000..448df6d4 --- /dev/null +++ b/botx/client/users_api/user_from_search.py @@ -0,0 +1,35 @@ +from typing import List, Literal, Optional +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.users import UserFromSearch + + +class BotXAPISearchUserResult(VerifiedPayloadBaseModel): + user_huid: UUID + ad_login: Optional[str] = None + ad_domain: Optional[str] = None + name: str + company: Optional[str] = None + company_position: Optional[str] = None + department: Optional[str] = None + emails: List[str] = Field(default_factory=list) + + +class BotXAPISearchUserResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPISearchUserResult + + def to_domain(self) -> UserFromSearch: + return UserFromSearch( + huid=self.result.user_huid, + ad_login=self.result.ad_login, + ad_domain=self.result.ad_domain, + username=self.result.name, + company=self.result.company, + company_position=self.result.company_position, + department=self.result.department, + emails=self.result.emails, + ) diff --git a/botx/clients/__init__.py b/botx/clients/__init__.py deleted file mode 100644 index 2434158d..00000000 --- a/botx/clients/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for BotX API related parts: requests, clients, types.""" diff --git a/botx/clients/clients/__init__.py b/botx/clients/clients/__init__.py deleted file mode 100644 index 7199c19d..00000000 --- a/botx/clients/clients/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for clients that make requests to BotX API.""" diff --git a/botx/clients/clients/async_client.py b/botx/clients/clients/async_client.py deleted file mode 100644 index 3d6bce71..00000000 --- a/botx/clients/clients/async_client.py +++ /dev/null @@ -1,145 +0,0 @@ -"""Definition for async client for BotX API.""" -from dataclasses import field -from http import HTTPStatus -from json import JSONDecodeError -from typing import Any, List, TypeVar - -import httpx -from pydantic.dataclasses import dataclass - -from botx.clients.clients.processing import extract_result, handle_error -from botx.clients.methods.base import BotXMethod -from botx.clients.types.http import ExpectedType, HTTPRequest, HTTPResponse -from botx.converters import optional_sequence_to_list -from botx.exceptions import ( - BotXAPIError, - BotXAPIRouteDeprecated, - BotXConnectError, - BotXJSONDecodeError, -) -from botx.shared import BotXDataclassConfig - -ResponseT = TypeVar("ResponseT") - - -@dataclass(config=BotXDataclassConfig) -class AsyncClient: - """Async client for BotX API.""" - - http_client: httpx.AsyncClient = field(init=False) - interceptors: List[Any] = field(default_factory=list) - - def __post_init__(self) -> None: - """Init or update special fields.""" - self.http_client = httpx.AsyncClient() - self.interceptors = optional_sequence_to_list(self.interceptors) - - # TODO: Use host as argument here. - @classmethod - def build_request(cls, method: BotXMethod[ResponseT]) -> HTTPRequest: - """Build HTTPRequest from passed BotX method. - - Arguments: - method: BotX method. - - Returns: - Built request. - """ - return method.build_http_request() - - async def process_response( - self, - method: BotXMethod[ResponseT], - response: HTTPResponse, - ) -> ResponseT: - """Handle errors and extract data from BotX API response. - - Arguments: - method: BotX API method. - response: HTTPResponse that is result of method executing. - - Returns: - Shape specified for method response. - - Raises: - BotXAPIError: raised if handler for error status code was not found. - BotXAPIRouteDeprecated: raised if API route was deprecated. - """ - handlers_dict = method.error_handlers - error_handlers = handlers_dict.get(response.status_code) - if error_handlers is not None: - await handle_error(method, error_handlers, response) - - if response.status_code == HTTPStatus.GONE: - raise BotXAPIRouteDeprecated( - url=method.url, - method=method.http_method, - status=response.status_code, - response_content=response.json_body, - ) - - if response.is_error or response.is_redirect: - raise BotXAPIError( - url=method.url, - method=method.http_method, - status=response.status_code, - response_content=response.json_body, - ) - - return extract_result(method, response) - - async def execute(self, request: HTTPRequest) -> HTTPResponse: - """Make request to BotX API. - - Arguments: - request: HTTPRequest that was built from method. - - Returns: - HTTP response from API. - - Raises: - BotXConnectError: raised if unable to connect to service. - BotXJSONDecodeError: raised if service returned invalid body. - """ - try: - response = await self.http_client.request( - request.method, - request.url, - headers=request.headers, - params=request.query_params, - json=request.json_body, - data=request.data, - files=request.files, - ) - except httpx.HTTPError as httpx_exc: - raise BotXConnectError( - url=request.url, - method=request.method, - ) from httpx_exc - - headers = dict(response.headers) - - should_process_as_error = ( - response.status_code in request.should_process_as_error - ) - if ( # noqa: WPS337 - not response.is_error - and not should_process_as_error # noqa: W503 - and request.expected_type == ExpectedType.BINARY # noqa: W503 - ): - return HTTPResponse( - headers=headers, - status_code=response.status_code, - raw_data=response.read(), - ) - - try: - json_body = response.json() - except JSONDecodeError as exc: - raise BotXJSONDecodeError(url=request.url, method=request.method) from exc - - return HTTPResponse( - headers=headers, - status_code=response.status_code, - json_body=json_body, - ) diff --git a/botx/clients/clients/processing.py b/botx/clients/clients/processing.py deleted file mode 100644 index 7c98b50d..00000000 --- a/botx/clients/clients/processing.py +++ /dev/null @@ -1,78 +0,0 @@ -"""Logic for handling response from BotX API for real HTTP responses.""" -import collections -import contextlib -from io import BytesIO -from typing import TypeVar - -from pydantic import ValidationError - -from botx import concurrency -from botx.clients.methods.base import APIResponse, BotXMethod, ErrorHandlersInMethod -from botx.clients.types.http import ExpectedType, HTTPResponse -from botx.models.files import File - -ResponseT = TypeVar("ResponseT") - - -def build_file(response: HTTPResponse) -> File: - """Build file from response raw data. - - Arguments: - response: HTTP response from BotX API. - - Returns: - Built file from response. - """ - mimetype = response.headers["content-type"].split(";", 1)[0] - ext = File.get_ext_by_mimetype(mimetype) or "" - file_name = "document{0}".format(ext) - return File.from_file(BytesIO(response.raw_data), file_name) # type: ignore - - -def extract_result( # noqa: WPS210 - method: BotXMethod[ResponseT], - response: HTTPResponse, -) -> ResponseT: - """Extract result from successful response and convert it to right shape. - - Arguments: - method: method to BotX API that was called. - response: HTTP response from BotX API. - - Returns: - Converted shape from BotX API. - """ - if method.expected_type == ExpectedType.BINARY: - return build_file(response) # type: ignore - - return_shape = method.returning - api_response = APIResponse[return_shape].parse_obj( # type: ignore - response.json_body, - ) - response_result = api_response.result - extractor = method.result_extractor - if extractor is not None: - # mypy does not understand that self passed here - return extractor(response_result) # type: ignore - - return response_result - - -async def handle_error( - method: BotXMethod, - error_handlers: ErrorHandlersInMethod, - response: HTTPResponse, -) -> None: - """Handle error status code from BotX API. - - Arguments: - method: method to BotX API that was called. - error_handlers: registered on method handlers for different responses. - response: HTTP response from BotX API. - """ - if not isinstance(error_handlers, collections.Sequence): - error_handlers = [error_handlers] - - for error_handler in error_handlers: - with contextlib.suppress(ValidationError): - await concurrency.callable_to_coroutine(error_handler, method, response) diff --git a/botx/clients/clients/sync_client.py b/botx/clients/clients/sync_client.py deleted file mode 100644 index a4ad37ed..00000000 --- a/botx/clients/clients/sync_client.py +++ /dev/null @@ -1,153 +0,0 @@ -"""Definition for sync client for BotX API.""" -from dataclasses import field -from http import HTTPStatus -from json import JSONDecodeError -from typing import Any, List, TypeVar - -import httpx -from pydantic.dataclasses import dataclass - -from botx import concurrency -from botx.clients.clients.processing import extract_result, handle_error -from botx.clients.methods.base import BotXMethod, ErrorHandlersInMethod -from botx.clients.types.http import ExpectedType, HTTPRequest, HTTPResponse -from botx.converters import optional_sequence_to_list -from botx.exceptions import ( - BotXAPIError, - BotXAPIRouteDeprecated, - BotXConnectError, - BotXJSONDecodeError, -) -from botx.shared import BotXDataclassConfig - -ResponseT = TypeVar("ResponseT") - - -@dataclass(config=BotXDataclassConfig) -class Client: - """Sync client for BotX API.""" - - http_client: httpx.Client = field(init=False) - interceptors: List[Any] = field(default_factory=list) - - def __post_init__(self) -> None: - """Init or update special fields.""" - self.http_client = httpx.Client() - self.interceptors = optional_sequence_to_list(self.interceptors) - - @classmethod - def build_request(cls, method: BotXMethod[ResponseT]) -> HTTPRequest: - """Build HTTPRequest from passed BotX method. - - Arguments: - method: BotX method. - - Returns: - Built request. - """ - return method.build_http_request() - - def process_response( - self, - method: BotXMethod[ResponseT], - response: HTTPResponse, - ) -> ResponseT: - """Handle errors and extract data from BotX API response. - - Arguments: - method: BotX API method. - response: HTTPResponse that is result of method executing. - - Returns: - Shape specified for method response. - - Raises: - BotXAPIError: raised if handler for error status code was not found. - BotXAPIRouteDeprecated: raised if API route was deprecated. - """ - handlers_dict = method.error_handlers - error_handlers = handlers_dict.get(response.status_code) - if error_handlers is not None: - _handle_error(method, error_handlers, response) - - if response.status_code == HTTPStatus.GONE: - raise BotXAPIRouteDeprecated( - url=method.url, - method=method.http_method, - status=response.status_code, - response_content=response.json_body, - ) - - if response.is_error or response.is_redirect: - raise BotXAPIError( - url=method.url, - method=method.http_method, - status=response.status_code, - response_content=response.json_body, - ) - - return extract_result(method, response) - - def execute(self, request: HTTPRequest) -> HTTPResponse: - """Make request to BotX API. - - Arguments: - request: HTTPRequest that was built from method. - - Returns: - HTTP response from API. - - Raises: - BotXConnectError: raised if unable to connect to service. - BotXJSONDecodeError: raised if service returned invalid body. - """ - try: - response = self.http_client.request( - request.method, - request.url, - headers=request.headers, - params=request.query_params, - json=request.json_body, - data=request.data, - files=request.files, - ) - except httpx.HTTPError as httpx_exc: - raise BotXConnectError( - url=request.url, - method=request.method, - ) from httpx_exc - - headers = dict(response.headers) - - should_process_as_error = ( - response.status_code in request.should_process_as_error - ) - if ( # noqa: WPS337 - not response.is_error - and not should_process_as_error # noqa: W503 - and request.expected_type == ExpectedType.BINARY # noqa: W503 - ): - return HTTPResponse( - headers=headers, - status_code=response.status_code, - raw_data=response.read(), - ) - - try: - json_body = response.json() - except JSONDecodeError as exc: - raise BotXJSONDecodeError(url=request.url, method=request.method) from exc - - return HTTPResponse( - headers=headers, - status_code=response.status_code, - json_body=json_body, - ) - - -def _handle_error( - method: BotXMethod, - error_handlers: ErrorHandlersInMethod, - response: HTTPResponse, -) -> None: - concurrency.async_to_sync(handle_error)(method, error_handlers, response) diff --git a/botx/clients/interceptors/__init__.py b/botx/clients/interceptors/__init__.py deleted file mode 100644 index 92a142d2..00000000 --- a/botx/clients/interceptors/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for requests interceptors for clietns.""" diff --git a/botx/clients/methods/__init__.py b/botx/clients/methods/__init__.py deleted file mode 100644 index 848781a1..00000000 --- a/botx/clients/methods/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for logic of requests to BotX API.""" diff --git a/botx/clients/methods/base.py b/botx/clients/methods/base.py deleted file mode 100644 index 74b35fef..00000000 --- a/botx/clients/methods/base.py +++ /dev/null @@ -1,231 +0,0 @@ -"""Definition for base class that is responsible for request to BotX API.""" -from __future__ import annotations - -import json -import typing -from abc import ABC, abstractmethod -from urllib.parse import urljoin - -from httpx import Response -from pydantic import BaseConfig, BaseModel, Extra -from pydantic.generics import GenericModel - -from botx.clients.types.http import ( - ExpectedType, - HTTPRequest, - HTTPResponse, - PrimitiveDataType, -) -from botx.models.enums import Statuses - -try: - from typing import Literal # noqa: WPS433, WPS458 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 - -ResponseT = typing.TypeVar("ResponseT") -SyncErrorHandler = typing.Callable[["BotXMethod", HTTPResponse], typing.NoReturn] -AsyncErrorHandler = typing.Callable[ - ["BotXMethod", Response], - typing.Awaitable[typing.NoReturn], -] -ErrorHandler = typing.Union[SyncErrorHandler, AsyncErrorHandler] -ErrorHandlersInMethod = typing.Union[typing.Sequence[ErrorHandler], ErrorHandler] - - -class APIResponse(GenericModel, typing.Generic[ResponseT]): - """Model for successful response from BotX API.""" - - #: status of requested operation response. - status: Literal[Statuses.ok] = Statuses.ok - - #: generic response shape. - result: ResponseT - - -class APIErrorResponse(GenericModel, typing.Generic[ResponseT]): - """Model for error response from BotX API.""" - - #: status of requested operation response. - status: Literal[Statuses.error] = Statuses.error - - #: reason why operation failed - reason: str - - #: errors from API. - errors: typing.List[str] - - #: additional payload with more data about error. - error_data: ResponseT - - -class AbstractBotXMethod(ABC, typing.Generic[ResponseT]): - """Abstract base class for BotX request.""" - - @property - @abstractmethod - def __url__(self) -> str: - """Path for method in BotX API.""" - - @property - @abstractmethod - def __method__(self) -> str: - """HTTP method used for method.""" - - @property - @abstractmethod - def __returning__(self) -> typing.Type[typing.Any]: - """Shape returned from method that can be parsed by pydantic.""" - - @property - def __errors_handlers__(self) -> typing.Mapping[int, ErrorHandlersInMethod]: - """Error handlers for responses from BotX API by status code and handler.""" - return typing.cast(typing.Mapping[int, ErrorHandlersInMethod], {}) - - @property - def __result_extractor__( - self, - ) -> typing.Optional[typing.Callable[[BotXMethod, typing.Any], ResponseT]]: - """Extractor for response shape from BotX API.""" - return None # noqa: WPS324 - - @property - def __expected_type__(self) -> ExpectedType: - """Extractor of expected type of response body.""" - return ExpectedType.JSON - - -CREDENTIALS_FIELDS = frozenset(("token", "host", "scheme")) - - -class BaseBotXMethod(AbstractBotXMethod[ResponseT], ABC): # noqa: WPS214 - """Base logic that is responsible for configuration and shortcuts for fields.""" - - #: host where request should be sent. - host: str = "" - - #: token for request. - token: str = "" - - #: HTTP scheme for request. - scheme: str = "https" - - @property - def url(self) -> str: - """Full URL for request.""" - base_url = "{scheme}://{host}".format(scheme=self.scheme, host=self.host) - return urljoin(base_url, self.__url__) - - @property - def http_method(self) -> str: - """HTTP method for request.""" - return self.__method__ - - @property - def headers(self) -> typing.Dict[str, str]: - """Headers that should be used in request.""" - return {"Content-Type": "application/json"} - - @property - def query_params(self) -> typing.Dict[str, PrimitiveDataType]: - """Query string query_params for request.""" - return {} - - @property - def returning(self) -> typing.Type[typing.Any]: - """Shape returned from method that can be parsed by pydantic.""" - return self.__returning__ - - @property - def error_handlers(self) -> typing.Mapping[int, ErrorHandlersInMethod]: - """Error handlers for responses from BotX API by status code and handler.""" - return self.__errors_handlers__ - - @property - def result_extractor( - self, - ) -> typing.Optional[typing.Callable[[BotXMethod, typing.Any], ResponseT]]: - """Extractor for response shape from BotX API.""" - return self.__result_extractor__ - - @property - def expected_type(self) -> ExpectedType: - """Extractor of expected type of response body.""" - return self.__expected_type__ - - -class BotXMethod(BaseBotXMethod[ResponseT], BaseModel, ABC): - """Method for BotX API that should be extended by actual implementation.""" - - class Config(BaseConfig): - extra = Extra.allow - allow_population_by_field_name = True - arbitrary_types_allowed = True - orm_mode = True - - def configure(self, *, host: str, token: str, scheme: str = "https") -> None: - """Configure request with credentials and transport related stuff. - - Arguments: - host: host where request should be sent. - token: token for request. - scheme: HTTP scheme for request. - """ - self.token = token - self.host = host - self.scheme = scheme - - def build_serialized_dict( - self, - ) -> typing.Optional[typing.Dict[str, PrimitiveDataType]]: - """Build serialized dict (with only primitive types) for request. - - Returns: - Serialized dict. - """ - # TODO: Waiting for - serialized_dict = json.loads( - self.json( - by_alias=True, - exclude=CREDENTIALS_FIELDS, - exclude_none=True, - ), - ) - - # Because exclude_none removes empty file key on message update - if hasattr(self, "file") and self.file is None: # type: ignore # noqa: WPS421 - serialized_dict["file"] = None - - return serialized_dict - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.query_params - request_data = self.build_serialized_dict() - - if self.__method__ == "GET" and request_data: - request_params = request_data - request_data = None - - return HTTPRequest( - method=self.__method__, - url=self.url, - headers=self.headers, - query_params=dict(request_params), - json_body=request_data, - ) - - -class AuthorizedBotXMethod(BotXMethod[ResponseT], ABC): - """Method for BotX API that adds authorization token.""" - - @property - def headers(self) -> typing.Dict[str, str]: - """Headers that should be used in request.""" - headers = super().headers - headers["Authorization"] = "Bearer {token}".format(token=self.token) - return headers diff --git a/botx/clients/methods/errors/__init__.py b/botx/clients/methods/errors/__init__.py deleted file mode 100644 index 86529f8e..00000000 --- a/botx/clients/methods/errors/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for built-in error handlers for responses from BotX API.""" diff --git a/botx/clients/methods/errors/bot_is_not_admin.py b/botx/clients/methods/errors/bot_is_not_admin.py deleted file mode 100644 index 1cde3a08..00000000 --- a/botx/clients/methods/errors/bot_is_not_admin.py +++ /dev/null @@ -1,54 +0,0 @@ -"""Definition for "bot is not admin" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class BotIsNotAdminError(BotXAPIError): - """Error for raising when bot is not admin.""" - - message_template = "bot {bot_id} is not admin of chat {group_chat_id}" - - #: ID of bot that sent request. - bot_id: UUID - - #: ID of chat into which request was sent. - group_chat_id: UUID - - -class BotIsNotAdminData(BaseModel): - """Data for error when bot is not admin.""" - - #: ID of sender (bot) - sender: UUID - - #: ID of chat into which request was sent. - group_chat_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "bot is not admin" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - BotIsNotAdminError: raised always. - """ - error_data = ( - APIErrorResponse[BotIsNotAdminData].parse_obj(response.json_body).error_data - ) - raise BotIsNotAdminError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - bot_id=error_data.sender, - group_chat_id=error_data.group_chat_id, - ) diff --git a/botx/clients/methods/errors/bot_not_found.py b/botx/clients/methods/errors/bot_not_found.py deleted file mode 100644 index ab08af3b..00000000 --- a/botx/clients/methods/errors/bot_not_found.py +++ /dev/null @@ -1,32 +0,0 @@ -"""Definition for "bot not found" error.""" -from typing import NoReturn - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class BotNotFoundError(BotXAPIError): - """Error for raising when bot not found.""" - - message_template = "bot with id `{bot_id}` not found. " - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "bot not found" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - BotNotFoundError: raised always. - """ - APIErrorResponse[dict].parse_obj(response.json_body) - raise BotNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - bot_id=method.bot_id, # type: ignore - ) diff --git a/botx/clients/methods/errors/chat_creation_disallowed.py b/botx/clients/methods/errors/chat_creation_disallowed.py deleted file mode 100644 index 080febc7..00000000 --- a/botx/clients/methods/errors/chat_creation_disallowed.py +++ /dev/null @@ -1,48 +0,0 @@ -"""Definition for "chat creating disallowed" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class ChatCreationDisallowedError(BotXAPIError): - """Error for raising when bot is not allowed to create chats.""" - - message_template = "bot {bot_id} is not allowed to create chats" - - #: ID of bot that sent request. - bot_id: UUID - - -class ChatCreationDisallowedData(BaseModel): - """Data for error when bot is not allowed to create chats.""" - - #: ID of bot that sent request. - bot_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "chat creating disallowed" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - ChatCreationDisallowedError: raised always. - """ - parsed_response = APIErrorResponse[ChatCreationDisallowedData].parse_obj( - response.json_body, - ) - error_data = parsed_response.error_data - raise ChatCreationDisallowedError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - bot_id=error_data.bot_id, - ) diff --git a/botx/clients/methods/errors/chat_creation_error.py b/botx/clients/methods/errors/chat_creation_error.py deleted file mode 100644 index 789f72eb..00000000 --- a/botx/clients/methods/errors/chat_creation_error.py +++ /dev/null @@ -1,31 +0,0 @@ -"""Definition for "chat creation error".""" -from typing import NoReturn - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class ChatCreationError(BotXAPIError): - """Error for raising when there is error for chat creation.""" - - message_template = "error while creating chat" - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "chat creation error" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - ChatCreationError: raised always. - """ - APIErrorResponse[dict].parse_obj(response.json_body) - raise ChatCreationError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - ) diff --git a/botx/clients/methods/errors/chat_is_not_modifiable.py b/botx/clients/methods/errors/chat_is_not_modifiable.py deleted file mode 100644 index 8b123378..00000000 --- a/botx/clients/methods/errors/chat_is_not_modifiable.py +++ /dev/null @@ -1,45 +0,0 @@ -"""Definition for "chat is not modifiable" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class PersonalChatIsNotModifiableError(BotXAPIError): - """Error for raising when chat is not modifiable.""" - - message_template = "personal chat is not modifiable" - - -class PersonalChatIsNotModifiableData(BaseModel): - """Data for error when chat is not modifiable.""" - - #: ID of chat that can not be modified. - group_chat_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "chat creation error" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - PersonalChatIsNotModifiableError: raised always. - """ - parsed_response = APIErrorResponse[PersonalChatIsNotModifiableData].parse_obj( - response.json_body, - ) - error_data = parsed_response.error_data - raise PersonalChatIsNotModifiableError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - group_chat_id=error_data.group_chat_id, - ) diff --git a/botx/clients/methods/errors/chat_not_found.py b/botx/clients/methods/errors/chat_not_found.py deleted file mode 100644 index 78544d69..00000000 --- a/botx/clients/methods/errors/chat_not_found.py +++ /dev/null @@ -1,47 +0,0 @@ -"""Definition for "chat not found" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class ChatNotFoundError(BotXAPIError): - """Error for raising when chat not found.""" - - message_template = "chat {group_chat_id} not found" - - #: ID of chat that was requested. - group_chat_id: UUID - - -class ChatNotFoundData(BaseModel): - """Data for error when chat not found.""" - - #: ID of chat that was requested. - group_chat_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "chat creation error" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - ChatNotFoundError: raised always. - """ - parsed_response = APIErrorResponse[ChatNotFoundData].parse_obj(response.json_body) - - error_data = parsed_response.error_data - raise ChatNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - group_chat_id=error_data.group_chat_id, - ) diff --git a/botx/clients/methods/errors/files/__init__.py b/botx/clients/methods/errors/files/__init__.py deleted file mode 100644 index 86529f8e..00000000 --- a/botx/clients/methods/errors/files/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for built-in error handlers for responses from BotX API.""" diff --git a/botx/clients/methods/errors/files/chat_not_found.py b/botx/clients/methods/errors/files/chat_not_found.py deleted file mode 100644 index 1858b48f..00000000 --- a/botx/clients/methods/errors/files/chat_not_found.py +++ /dev/null @@ -1,53 +0,0 @@ -"""Definition for "chat not found" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class ChatNotFoundError(BotXAPIError): - """Error for raising when chat not found.""" - - message_template = "{error_description}" - - #: description of error. - error_description: str - - -class ChatNotFoundData(BaseModel): - """Data for error when chat not found.""" - - #: ID of chat where file is from. - group_chat_id: UUID - - #: description of error. - error_description: str - - class Config: - extra = "forbid" - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "chat not found" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - ChatNotFoundError: raised always. - """ - parsed_response = APIErrorResponse[ChatNotFoundData].parse_obj(response.json_body) - - error_data = parsed_response.error_data - raise ChatNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - error_description=error_data.error_description, - ) diff --git a/botx/clients/methods/errors/files/file_deleted.py b/botx/clients/methods/errors/files/file_deleted.py deleted file mode 100644 index 8f3216db..00000000 --- a/botx/clients/methods/errors/files/file_deleted.py +++ /dev/null @@ -1,51 +0,0 @@ -"""Definition for "file deleted" error.""" -from typing import NoReturn - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class FileDeletedError(BotXAPIError): - """Error for raising when file deleted.""" - - message_template = "{error_description}" - - #: description of error. - error_description: str - - -class FileDeletedErrorData(BaseModel): - """Data for error when file deleted.""" - - #: link of deleted file. - link: str - - #: description of error. - error_description: str - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "file deleted" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - FileDeletedError: raised always. - """ - parsed_response = APIErrorResponse[FileDeletedErrorData].parse_obj( - response.json_body, - ) - - error_data = parsed_response.error_data - raise FileDeletedError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - error_description=error_data.error_description, - ) diff --git a/botx/clients/methods/errors/files/metadata_not_found.py b/botx/clients/methods/errors/files/metadata_not_found.py deleted file mode 100644 index f1239583..00000000 --- a/botx/clients/methods/errors/files/metadata_not_found.py +++ /dev/null @@ -1,62 +0,0 @@ -"""Definition for "file metadata not found" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class MetadataNotFoundError(BotXAPIError): - """Error for raising when file metadata not found.""" - - message_template = ( - "File with specified file_id `{file_id}` and " - "group_chat_id `{group_chat_id}` not found in file service." - ) - - #: ID of file which metadata was requested. - file_id: UUID - - #: ID of chat where file is from. - group_chat_id: UUID - - -class MetadataNotFoundData(BaseModel): - """Data for error when file metadata not found.""" - - #: ID of file which metadata was requested. - file_id: UUID - - #: ID of chat where file is from. - group_chat_id: UUID - - #: description of error. - error_description: str - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "file metadata not found" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - MetadataNotFoundError: raised always. - """ - parsed_response = APIErrorResponse[MetadataNotFoundData].parse_obj( - response.json_body, - ) - - error_data = parsed_response.error_data - raise MetadataNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - file_id=error_data.file_id, - group_chat_id=error_data.group_chat_id, - ) diff --git a/botx/clients/methods/errors/files/without_preview.py b/botx/clients/methods/errors/files/without_preview.py deleted file mode 100644 index e78c478e..00000000 --- a/botx/clients/methods/errors/files/without_preview.py +++ /dev/null @@ -1,53 +0,0 @@ -"""Definition for "without preview" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class WithoutPreviewError(BotXAPIError): - """Error for raising when there is no file preview.""" - - message_template = "{error_description}" - - #: description of error. - error_description: str - - -class WithoutPreviewData(BaseModel): - """Data for error when there is no file preview.""" - - #: ID of file which preview was requested. - file_id: UUID - - #: ID of chat where file is from. - group_chat_id: UUID - - #: description of error. - error_description: str - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "without preview" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - WithoutPreviewError: raised always. - """ - parsed_response = APIErrorResponse[WithoutPreviewData].parse_obj(response.json_body) - - error_data = parsed_response.error_data - raise WithoutPreviewError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - error_description=error_data.error_description, - ) diff --git a/botx/clients/methods/errors/messaging.py b/botx/clients/methods/errors/messaging.py deleted file mode 100644 index 03d806fb..00000000 --- a/botx/clients/methods/errors/messaging.py +++ /dev/null @@ -1,30 +0,0 @@ -"""Definition for "messaging" error.""" -from typing import NoReturn - -from botx.clients.methods.base import BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class MessagingError(BotXAPIError): - """Error for raising when there is messaging error.""" - - message_template = "error from messaging service" - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle messaging error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - MessagingError: raised always. - """ - raise MessagingError( - url=method.url, - method=method.http_method, - response=response.json_body, - status=response.status_code, - ) diff --git a/botx/clients/methods/errors/permissions.py b/botx/clients/methods/errors/permissions.py deleted file mode 100644 index 0a37b99a..00000000 --- a/botx/clients/methods/errors/permissions.py +++ /dev/null @@ -1,51 +0,0 @@ -"""Definition for "no permission" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class NoPermissionError(BotXAPIError): - """Error for raising when there is no permission for operation.""" - - message_template = ( - "Bot doesn't have permission for this operation in chat {group_chat_id}" - ) - - #: ID of chat that was requested. - group_chat_id: UUID - - -class NoPermissionErrorData(BaseModel): - """Data for error when there is no permission for operation.""" - - #: ID of chat that was requested. - group_chat_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "no permission" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - NoPermissionError: raised always. - """ - parsed_response = APIErrorResponse[NoPermissionErrorData].parse_obj( - response.json_body, - ) - - error_data = parsed_response.error_data - raise NoPermissionError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - group_chat_id=error_data.group_chat_id, - ) diff --git a/botx/clients/methods/errors/stickers/__init__.py b/botx/clients/methods/errors/stickers/__init__.py deleted file mode 100644 index 86529f8e..00000000 --- a/botx/clients/methods/errors/stickers/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for built-in error handlers for responses from BotX API.""" diff --git a/botx/clients/methods/errors/stickers/image_not_valid.py b/botx/clients/methods/errors/stickers/image_not_valid.py deleted file mode 100644 index ecf6c191..00000000 --- a/botx/clients/methods/errors/stickers/image_not_valid.py +++ /dev/null @@ -1,30 +0,0 @@ -"""Definition for "image is not valid" error.""" -from typing import NoReturn - -from botx.clients.methods.base import BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class ImageNotValidError(BotXAPIError): - """Error for raising when image is not valid.""" - - message_template = "image is not valid" - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "image is not valid" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - ImageNotValidError: raised always. - """ - raise ImageNotValidError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - ) diff --git a/botx/clients/methods/errors/stickers/sticker_pack_not_found.py b/botx/clients/methods/errors/stickers/sticker_pack_not_found.py deleted file mode 100644 index 3e593eca..00000000 --- a/botx/clients/methods/errors/stickers/sticker_pack_not_found.py +++ /dev/null @@ -1,49 +0,0 @@ -"""Definition for "sticker pack was not found" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class StickerPackNotFoundError(BotXAPIError): - """Error for raising when sticker pack was not found.""" - - message_template = "sticker pack {pack_id} not found" - - #: sticker pack ID. - pack_id: UUID - - -class StickerPackNotFoundData(BaseModel): - """Data for error when sticker pack was not found.""" - - #: sticker pack ID. - pack_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "sticker pack getting error" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - StickerPackNotFoundError: raised always. - """ - parsed_response = APIErrorResponse[StickerPackNotFoundData].parse_obj( - response.json_body, - ) - - error_data = parsed_response.error_data - raise StickerPackNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - pack_id=error_data.pack_id, - ) diff --git a/botx/clients/methods/errors/stickers/sticker_pack_or_sticker_not_found.py b/botx/clients/methods/errors/stickers/sticker_pack_or_sticker_not_found.py deleted file mode 100644 index b6e354da..00000000 --- a/botx/clients/methods/errors/stickers/sticker_pack_or_sticker_not_found.py +++ /dev/null @@ -1,56 +0,0 @@ -"""Definition for "sticker pack or sticker was not found" error.""" -from typing import NoReturn -from uuid import UUID - -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class StickerPackOrStickerNotFoundError(BotXAPIError): - """Error for raising when sticker pack or sticker was not found.""" - - message_template = "sticker pack {pack_id} or sticker {sticker_id} was not found" - - #: sticker pack ID. - pack_id: UUID - - #: sticker ID. - sticker_id: UUID - - -class StickerPackOrStickerNotFoundData(BaseModel): - """Data for error when sticker pack or sticker was not found.""" - - #: sticker pack ID. - pack_id: UUID - - #: sticker ID. - sticker_id: UUID - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "sticker getting error" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - StickerPackOrStickerNotFoundError: raised always. - """ - parsed_response = APIErrorResponse[StickerPackOrStickerNotFoundData].parse_obj( - response.json_body, - ) - - error_data = parsed_response.error_data - raise StickerPackOrStickerNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - pack_id=error_data.pack_id, - sticker_id=error_data.sticker_id, - ) diff --git a/botx/clients/methods/errors/unauthorized_bot.py b/botx/clients/methods/errors/unauthorized_bot.py deleted file mode 100644 index a956dee6..00000000 --- a/botx/clients/methods/errors/unauthorized_bot.py +++ /dev/null @@ -1,34 +0,0 @@ -"""Definition for "invalid bot credentials" error.""" -from typing import NoReturn - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class InvalidBotCredentials(BotXAPIError): - """Error for raising when got invalid bot credentials.""" - - message_template = ( - "Can't get token for bot {bot_id}. Make sure bot credentials is correct" - ) - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "invalid bot credentials" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - InvalidBotCredentials: raised always. - """ - APIErrorResponse[dict].parse_obj(response.json_body) - raise InvalidBotCredentials( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - bot_id=method.bot_id, # type: ignore - ) diff --git a/botx/clients/methods/errors/user_not_found.py b/botx/clients/methods/errors/user_not_found.py deleted file mode 100644 index e146bae8..00000000 --- a/botx/clients/methods/errors/user_not_found.py +++ /dev/null @@ -1,31 +0,0 @@ -"""Definition for "user not found" error.""" -from typing import NoReturn - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.clients.types.http import HTTPResponse -from botx.exceptions import BotXAPIError - - -class UserNotFoundError(BotXAPIError): - """Error for raising when user not found.""" - - message_template = "user not found" - - -def handle_error(method: BotXMethod, response: HTTPResponse) -> NoReturn: - """Handle "user not found" error response. - - Arguments: - method: method which was made before error. - response: HTTP response from BotX API. - - Raises: - UserNotFoundError: raised always. - """ - APIErrorResponse[dict].parse_obj(response.json_body) - raise UserNotFoundError( - url=method.url, - method=method.http_method, - response_content=response.json_body, - status_content=response.status_code, - ) diff --git a/botx/clients/methods/extractors.py b/botx/clients/methods/extractors.py deleted file mode 100644 index a7408d5a..00000000 --- a/botx/clients/methods/extractors.py +++ /dev/null @@ -1,31 +0,0 @@ -"""Custom extractors for responses from BotX API.""" -from uuid import UUID - -from botx.clients.methods.base import BotXMethod -from botx.clients.types.response_results import ChatCreatedResult, PushResult - - -def extract_generated_sync_id(_method: BotXMethod, push: PushResult) -> UUID: - """Extract generated sync ID from response. - - Arguments: - _method: method that was used for making request. - push: push response from BotX API for generated message. - - Returns: - Extracted sync ID. - """ - return push.sync_id - - -def extract_generated_chat_id(_method: BotXMethod, response: ChatCreatedResult) -> UUID: - """Extract generated sync ID from response. - - Arguments: - _method: method that was used for making request. - response: response for created chat. - - Returns: - Extracted chat ID. - """ - return response.chat_id diff --git a/botx/clients/methods/v2/__init__.py b/botx/clients/methods/v2/__init__.py deleted file mode 100644 index ecf8b531..00000000 --- a/botx/clients/methods/v2/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for V3 API methods for BotX API.""" diff --git a/botx/clients/methods/v2/bots/__init__.py b/botx/clients/methods/v2/bots/__init__.py deleted file mode 100644 index 4d34b3c7..00000000 --- a/botx/clients/methods/v2/bots/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for bots resource.""" diff --git a/botx/clients/methods/v2/bots/token.py b/botx/clients/methods/v2/bots/token.py deleted file mode 100644 index 624411e8..00000000 --- a/botx/clients/methods/v2/bots/token.py +++ /dev/null @@ -1,45 +0,0 @@ -"""Method for retrieving token for bot.""" -from http import HTTPStatus -from typing import Dict -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import BotXMethod, PrimitiveDataType -from botx.clients.methods.errors import bot_not_found, unauthorized_bot - - -class Token(BotXMethod[str]): - """Method for retrieving token for bot.""" - - __url__ = "/api/v2/botx/bots/{bot_id}/token" - __method__ = "GET" - __returning__ = str - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: bot_not_found.handle_error, - HTTPStatus.UNAUTHORIZED: unauthorized_bot.handle_error, - } - - #: ID of bot which access for token. - bot_id: UUID - - #: calculated signature from secret_key for bot. - signature: str - - @property - def url(self) -> str: - """Full URL for request with filling bot_id.""" - api_url = self.__url__.format(bot_id=self.bot_id) - return urljoin(super().url, api_url) - - @property - def query_params(self) -> Dict[str, PrimitiveDataType]: - """Query string query_params for request with signature key.""" - return {"signature": self.signature} - - def build_serialized_dict(self) -> None: - """Return nothing to override dict body. - - Returns: - Nothing. - """ - return None # noqa: WPS324 diff --git a/botx/clients/methods/v3/__init__.py b/botx/clients/methods/v3/__init__.py deleted file mode 100644 index ecf8b531..00000000 --- a/botx/clients/methods/v3/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for V3 API methods for BotX API.""" diff --git a/botx/clients/methods/v3/chats/__init__.py b/botx/clients/methods/v3/chats/__init__.py deleted file mode 100644 index 9acbfb6f..00000000 --- a/botx/clients/methods/v3/chats/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for chat resource.""" diff --git a/botx/clients/methods/v3/chats/add_admin_role.py b/botx/clients/methods/v3/chats/add_admin_role.py deleted file mode 100644 index 1d587c42..00000000 --- a/botx/clients/methods/v3/chats/add_admin_role.py +++ /dev/null @@ -1,32 +0,0 @@ -"""Method for promoting users to admins.""" -from http import HTTPStatus -from typing import List -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import ( - bot_is_not_admin, - chat_is_not_modifiable, - chat_not_found, -) - - -class AddAdminRole(AuthorizedBotXMethod[bool]): - """Method for promoting users to chat admins.""" - - __url__ = "/api/v3/botx/chats/add_admin" - __method__ = "POST" - __returning__ = bool - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: ( - bot_is_not_admin.handle_error, - chat_is_not_modifiable.handle_error, - ), - HTTPStatus.NOT_FOUND: (chat_not_found.handle_error,), - } - - #: ID of chat where action should be performed. - group_chat_id: UUID - - #: IDs of users that should be promoted to admins. - user_huids: List[UUID] diff --git a/botx/clients/methods/v3/chats/add_user.py b/botx/clients/methods/v3/chats/add_user.py deleted file mode 100644 index 5b60c521..00000000 --- a/botx/clients/methods/v3/chats/add_user.py +++ /dev/null @@ -1,32 +0,0 @@ -"""Method for adding new users into chat.""" -from http import HTTPStatus -from typing import List -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import ( - bot_is_not_admin, - chat_is_not_modifiable, - chat_not_found, -) - - -class AddUser(AuthorizedBotXMethod[bool]): - """Method for adding new users into chat.""" - - __url__ = "/api/v3/botx/chats/add_user" - __method__ = "POST" - __returning__ = bool - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: ( - bot_is_not_admin.handle_error, - chat_is_not_modifiable.handle_error, - ), - HTTPStatus.NOT_FOUND: (chat_not_found.handle_error,), - } - - #: ID of chat into which users should be added. - group_chat_id: UUID - - #: IDs of users that should be added into chat. - user_huids: List[UUID] diff --git a/botx/clients/methods/v3/chats/chat_list.py b/botx/clients/methods/v3/chats/chat_list.py deleted file mode 100644 index 319ca358..00000000 --- a/botx/clients/methods/v3/chats/chat_list.py +++ /dev/null @@ -1,12 +0,0 @@ -"""Method for retrieving information about chat.""" - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.models.chats import BotChatList - - -class ChatList(AuthorizedBotXMethod[BotChatList]): # noqa: WPS110 - """Method for retrieving list of bot's chats.""" - - __url__ = "/api/v3/botx/chats/list" - __method__ = "GET" - __returning__ = BotChatList diff --git a/botx/clients/methods/v3/chats/create.py b/botx/clients/methods/v3/chats/create.py deleted file mode 100644 index f1c562aa..00000000 --- a/botx/clients/methods/v3/chats/create.py +++ /dev/null @@ -1,42 +0,0 @@ -"""Method for creating new chat.""" - -from http import HTTPStatus -from typing import List, Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import chat_creation_disallowed, chat_creation_error -from botx.clients.methods.extractors import extract_generated_chat_id -from botx.clients.types.response_results import ChatCreatedResult -from botx.models.enums import ChatTypes - - -class Create(AuthorizedBotXMethod[UUID]): - """Method for creating new chat.""" - - __url__ = "/api/v3/botx/chats/create" - __method__ = "POST" - __returning__ = ChatCreatedResult - __result_extractor__ = extract_generated_chat_id - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: chat_creation_disallowed.handle_error, - HTTPStatus.UNPROCESSABLE_ENTITY: chat_creation_error.handle_error, - } - - #: name of chat that should be created. - name: str - - #: description of new chat. - description: Optional[str] = None - - #: HUIDs of users that should be added into chat. - members: List[UUID] - - #: logo image of chat. - avatar: Optional[str] = None - - #: chat type. - chat_type: ChatTypes - - #: chat history is available to newcomers. - shared_history: bool diff --git a/botx/clients/methods/v3/chats/info.py b/botx/clients/methods/v3/chats/info.py deleted file mode 100644 index dd96e8f5..00000000 --- a/botx/clients/methods/v3/chats/info.py +++ /dev/null @@ -1,19 +0,0 @@ -"""Method for retrieving information about chat.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import messaging -from botx.models.chats import ChatFromSearch - - -class Info(AuthorizedBotXMethod[ChatFromSearch]): # noqa: WPS110 - """Method for retrieving information about chat.""" - - __url__ = "/api/v3/botx/chats/info" - __method__ = "GET" - __returning__ = ChatFromSearch - __errors_handlers__ = {HTTPStatus.BAD_REQUEST: messaging.handle_error} - - #: ID of chat for about which information should be retrieving. - group_chat_id: UUID diff --git a/botx/clients/methods/v3/chats/pin_message.py b/botx/clients/methods/v3/chats/pin_message.py deleted file mode 100644 index 43ac73eb..00000000 --- a/botx/clients/methods/v3/chats/pin_message.py +++ /dev/null @@ -1,24 +0,0 @@ -"""Method for pinning message in chat.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import chat_not_found, permissions - - -class PinMessage(AuthorizedBotXMethod[str]): - """Method for pinning message in chat.""" - - __url__ = "/api/v3/botx/chats/pin_message" - __method__ = "POST" - __returning__ = str - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: chat_not_found.handle_error, - HTTPStatus.FORBIDDEN: permissions.handle_error, - } - - #: ID of chat where message should be pinned. - chat_id: UUID - - #: ID of message that should be pinned. - sync_id: UUID diff --git a/botx/clients/methods/v3/chats/remove_user.py b/botx/clients/methods/v3/chats/remove_user.py deleted file mode 100644 index 641055ef..00000000 --- a/botx/clients/methods/v3/chats/remove_user.py +++ /dev/null @@ -1,32 +0,0 @@ -"""Method for removing users from chat.""" -from http import HTTPStatus -from typing import List -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import ( - bot_is_not_admin, - chat_is_not_modifiable, - chat_not_found, -) - - -class RemoveUser(AuthorizedBotXMethod[bool]): - """Method for removing users from chat.""" - - __url__ = "/api/v3/botx/chats/remove_user" - __method__ = "POST" - __returning__ = bool - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: ( - bot_is_not_admin.handle_error, - chat_is_not_modifiable.handle_error, - ), - HTTPStatus.NOT_FOUND: (chat_not_found.handle_error,), - } - - #: ID of chat from which users should be removed. - group_chat_id: UUID - - #: HUID of users that should be removed. - user_huids: List[UUID] diff --git a/botx/clients/methods/v3/chats/stealth_disable.py b/botx/clients/methods/v3/chats/stealth_disable.py deleted file mode 100644 index 321eeacd..00000000 --- a/botx/clients/methods/v3/chats/stealth_disable.py +++ /dev/null @@ -1,21 +0,0 @@ -"""Method for disabling stealth in chat.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import bot_is_not_admin, chat_not_found - - -class StealthDisable(AuthorizedBotXMethod[bool]): - """Method for disabling stealth in chat.""" - - __url__ = "/api/v3/botx/chats/stealth_disable" - __method__ = "POST" - __returning__ = bool - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: bot_is_not_admin.handle_error, - HTTPStatus.NOT_FOUND: chat_not_found.handle_error, - } - - #: ID of chat where stealth should be disabled. - group_chat_id: UUID diff --git a/botx/clients/methods/v3/chats/stealth_set.py b/botx/clients/methods/v3/chats/stealth_set.py deleted file mode 100644 index 93b26841..00000000 --- a/botx/clients/methods/v3/chats/stealth_set.py +++ /dev/null @@ -1,31 +0,0 @@ -"""Method for enabling stealth in chat.""" -from http import HTTPStatus -from typing import Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import bot_is_not_admin, chat_not_found - - -class StealthSet(AuthorizedBotXMethod[bool]): - """Method for enabling stealth in chat.""" - - __url__ = "/api/v3/botx/chats/stealth_set" - __method__ = "POST" - __returning__ = bool - __errors_handlers__ = { - HTTPStatus.FORBIDDEN: bot_is_not_admin.handle_error, - HTTPStatus.NOT_FOUND: chat_not_found.handle_error, - } - - #: ID of chat where stealth should be enabled. - group_chat_id: UUID - - #: should messages be shown in web. - disable_web: bool = False - - #: time of messages burning after read. - burn_in: Optional[int] = None - - #: time of messages burning after send. - expire_in: Optional[int] = None diff --git a/botx/clients/methods/v3/chats/unpin_message.py b/botx/clients/methods/v3/chats/unpin_message.py deleted file mode 100644 index 09ca4796..00000000 --- a/botx/clients/methods/v3/chats/unpin_message.py +++ /dev/null @@ -1,21 +0,0 @@ -"""Method for unpinning message in chat.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import chat_not_found, permissions - - -class UnpinMessage(AuthorizedBotXMethod[str]): - """Method for unpinning message in chat.""" - - __url__ = "/api/v3/botx/chats/unpin_message" - __method__ = "POST" - __returning__ = str - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: chat_not_found.handle_error, - HTTPStatus.FORBIDDEN: permissions.handle_error, - } - - #: ID of chat where message should be unpinned. - chat_id: UUID diff --git a/botx/clients/methods/v3/command/__init__.py b/botx/clients/methods/v3/command/__init__.py deleted file mode 100644 index 1c917907..00000000 --- a/botx/clients/methods/v3/command/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for command resource.""" diff --git a/botx/clients/methods/v3/command/command_result.py b/botx/clients/methods/v3/command/command_result.py deleted file mode 100644 index 5132db72..00000000 --- a/botx/clients/methods/v3/command/command_result.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Method for sending command result into chat.""" - -from typing import Optional -from uuid import UUID - -from pydantic import Field - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.extractors import extract_generated_sync_id -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.clients.types.response_results import PushResult -from botx.models.files import File -from botx.models.typing import AvailableRecipients - - -class CommandResult(AuthorizedBotXMethod[UUID]): - """Method for sending notification into many chats.""" - - __url__ = "/api/v3/botx/command/callback" - __method__ = "POST" - __returning__ = PushResult - __result_extractor__ = extract_generated_sync_id - - #: ID of event on which this answer will be sent. - sync_id: UUID - - #: custom ID of new message. - event_sync_id: Optional[UUID] = None - - #: users that will receive message. - recipients: AvailableRecipients = "all" - - #: message payload. - result: ResultPayload = Field(..., alias="command_result") - - #: attached file. - file: Optional[File] = None - - #: extra options for new message. - opts: ResultOptions = ResultOptions() diff --git a/botx/clients/methods/v3/events/__init__.py b/botx/clients/methods/v3/events/__init__.py deleted file mode 100644 index f7f18591..00000000 --- a/botx/clients/methods/v3/events/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for events resource.""" diff --git a/botx/clients/methods/v3/events/edit_event.py b/botx/clients/methods/v3/events/edit_event.py deleted file mode 100644 index 25ce6a3f..00000000 --- a/botx/clients/methods/v3/events/edit_event.py +++ /dev/null @@ -1,30 +0,0 @@ -"""Method for editing sent event.""" -from typing import Optional -from uuid import UUID - -from pydantic import Field - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.types.message_payload import UpdatePayload -from botx.clients.types.options import ResultOptions -from botx.models.files import File - - -class EditEvent(AuthorizedBotXMethod[str]): - """Method for editing sent event.""" - - __url__ = "/api/v3/botx/events/edit_event" - __method__ = "POST" - __returning__ = str - - #: ID of event that should be edited. - sync_id: UUID - - #: data for editing. - result: UpdatePayload = Field(..., alias="payload") - - #: file attached to message. - file: Optional[File] = None - - #: extra options for message. - opts: ResultOptions = ResultOptions() diff --git a/botx/clients/methods/v3/events/reply_event.py b/botx/clients/methods/v3/events/reply_event.py deleted file mode 100644 index b5717a99..00000000 --- a/botx/clients/methods/v3/events/reply_event.py +++ /dev/null @@ -1,31 +0,0 @@ -"""Method for sending command result into chat.""" - -from typing import Optional -from uuid import UUID - -from pydantic import Field - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.models.files import File - - -class ReplyEvent(AuthorizedBotXMethod[UUID]): - """Method for sending reply message.""" - - __url__ = "/api/v3/botx/events/reply_event" - __method__ = "POST" - __returning__ = str - - #: ID of message for reply. - source_sync_id: Optional[UUID] = None - - #: message payload. - result: ResultPayload = Field(..., alias="reply") - - #: attached file. - file: Optional[File] = None - - #: extra options for new message. - opts: ResultOptions = ResultOptions() diff --git a/botx/clients/methods/v3/files/__init__.py b/botx/clients/methods/v3/files/__init__.py deleted file mode 100644 index 2ad05f3b..00000000 --- a/botx/clients/methods/v3/files/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for files resource.""" diff --git a/botx/clients/methods/v3/files/download.py b/botx/clients/methods/v3/files/download.py deleted file mode 100644 index a9a8a0c0..00000000 --- a/botx/clients/methods/v3/files/download.py +++ /dev/null @@ -1,57 +0,0 @@ -"""Method for downloading file from chat.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.files import ( - chat_not_found, - file_deleted, - metadata_not_found, - without_preview, -) -from botx.clients.types.http import ExpectedType, HTTPRequest -from botx.models.files import File - - -class DownloadFile(AuthorizedBotXMethod[File]): - """Method for downloading a file from chat.""" - - __url__ = "/api/v3/botx/files/download" - __method__ = "GET" - __returning__ = File - __expected_type__ = ExpectedType.BINARY - __errors_handlers__ = { - HTTPStatus.NO_CONTENT: (file_deleted.handle_error,), - HTTPStatus.BAD_REQUEST: (without_preview.handle_error,), - HTTPStatus.NOT_FOUND: ( - chat_not_found.handle_error, - metadata_not_found.handle_error, - ), - } - - #: ID of the chat with file. - group_chat_id: UUID - - #: ID of the file for downloading. - file_id: UUID - - #: preview or file. - is_preview: bool - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=dict(request_params), # type: ignore - json_body={}, - expected_type=self.expected_type, - should_process_as_error=[HTTPStatus.NO_CONTENT], - ) diff --git a/botx/clients/methods/v3/files/upload.py b/botx/clients/methods/v3/files/upload.py deleted file mode 100644 index 7f5e7a41..00000000 --- a/botx/clients/methods/v3/files/upload.py +++ /dev/null @@ -1,60 +0,0 @@ -"""Method for uploading file to chat.""" -from http import HTTPStatus -from typing import Dict -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.files import chat_not_found -from botx.clients.types.http import HTTPRequest -from botx.clients.types.upload_file import UploadingFileMeta -from botx.models.files import File, MetaFile - - -class UploadFile(AuthorizedBotXMethod[MetaFile]): - """Method for uploading file to a chat.""" - - __url__ = "/api/v3/botx/files/upload" - __method__ = "POST" - __returning__ = MetaFile - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (chat_not_found.handle_error,), - } - - #: ID of the chat. - group_chat_id: UUID - - #: file for uploading. - file: File - - #: file metadata. - meta: UploadingFileMeta - - @property - def headers(self) -> Dict[str, str]: - """Headers that should be used in request.""" - headers = super().headers - # used to enable the client to attach a Content-Type with a boundary - headers.pop("Content-Type") - return headers - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - files = {"content": (self.file.file_name, self.file.file)} - request_data = { - "group_chat_id": str(self.group_chat_id), - "meta": self.meta.json(), - } - - return HTTPRequest( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=self.query_params, - json_body={}, - data=request_data, - files=files, # type: ignore - ) diff --git a/botx/clients/methods/v3/notification/__init__.py b/botx/clients/methods/v3/notification/__init__.py deleted file mode 100644 index dabef1cb..00000000 --- a/botx/clients/methods/v3/notification/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for notification resource.""" diff --git a/botx/clients/methods/v3/notification/direct_notification.py b/botx/clients/methods/v3/notification/direct_notification.py deleted file mode 100644 index 6e3af91f..00000000 --- a/botx/clients/methods/v3/notification/direct_notification.py +++ /dev/null @@ -1,40 +0,0 @@ -"""Method for sending notification into single chat.""" -from typing import Optional -from uuid import UUID - -from pydantic import Field - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.extractors import extract_generated_sync_id -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.clients.types.response_results import PushResult -from botx.models.files import File -from botx.models.typing import AvailableRecipients - - -class NotificationDirect(AuthorizedBotXMethod[UUID]): - """Method for sending notification into single chat.""" - - __url__ = "/api/v3/botx/notification/callback/direct" - __method__ = "POST" - __returning__ = PushResult - __result_extractor__ = extract_generated_sync_id - - #: ID of chat for new notification. - group_chat_id: UUID - - #: custom ID for message. - event_sync_id: Optional[UUID] = None - - #: HUIDs of users that should receive notifications. - recipients: AvailableRecipients = "all" - - #: data for build message: body, markup, mentions. - result: ResultPayload = Field(..., alias="notification") - - #: attached file for message. - file: Optional[File] = None - - #: extra options for message. - opts: ResultOptions = ResultOptions() diff --git a/botx/clients/methods/v3/notification/notification.py b/botx/clients/methods/v3/notification/notification.py deleted file mode 100644 index 1076d683..00000000 --- a/botx/clients/methods/v3/notification/notification.py +++ /dev/null @@ -1,34 +0,0 @@ -"""Method for sending notification into many chats.""" -from typing import List, Optional -from uuid import UUID - -from pydantic import Field - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.types.message_payload import ResultPayload -from botx.clients.types.options import ResultOptions -from botx.models.files import File -from botx.models.typing import AvailableRecipients - - -class Notification(AuthorizedBotXMethod[str]): - """Method for sending notification into many chats.""" - - __url__ = "/api/v3/botx/notification/callback" - __method__ = "POST" - __returning__ = str - - #: IDs of chats for new notification. - group_chat_ids: List[UUID] = [] - - #: HUIDs of users that should receive notifications. - recipients: AvailableRecipients = "all" - - #: data for build message: body, markup, mentions. - result: ResultPayload = Field(..., alias="notification") - - #: attached file for message. - file: Optional[File] = None - - #: extra options for message. - opts: ResultOptions = ResultOptions() diff --git a/botx/clients/methods/v3/smartapps/__init__.py b/botx/clients/methods/v3/smartapps/__init__.py deleted file mode 100644 index 1e1bdd78..00000000 --- a/botx/clients/methods/v3/smartapps/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for smartapp methods.""" diff --git a/botx/clients/methods/v3/smartapps/smartapp_event.py b/botx/clients/methods/v3/smartapps/smartapp_event.py deleted file mode 100644 index c4a0e1c0..00000000 --- a/botx/clients/methods/v3/smartapps/smartapp_event.py +++ /dev/null @@ -1,38 +0,0 @@ -"""Method for sending smartapp event.""" -from typing import Any, Dict, List, Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.models.files import File, MetaFile - - -class SmartAppEvent(AuthorizedBotXMethod[str]): - """Method for sending smartapp events.""" - - __url__ = "/api/v3/botx/smartapps/event" - __method__ = "POST" - __returning__ = str - - #: unique request id - ref: Optional[UUID] = None - - #: smartapp id - smartapp_id: UUID - - #: event data - data: Dict[str, Any] # noqa: WPS110 - - #: event options - opts: Dict[str, Any] = {} - - #: version of protocol smartapp <-> bot - smartapp_api_version: int - - #: smartapp chat - group_chat_id: Optional[UUID] - - #: files - files: List[File] = [] - - #: file's meta to upload - async_files: List[MetaFile] = [] diff --git a/botx/clients/methods/v3/smartapps/smartapp_notification.py b/botx/clients/methods/v3/smartapps/smartapp_notification.py deleted file mode 100644 index 574beddb..00000000 --- a/botx/clients/methods/v3/smartapps/smartapp_notification.py +++ /dev/null @@ -1,25 +0,0 @@ -"""Method for sending smartapp event.""" -from typing import Any, Dict, Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod - - -class SmartAppNotification(AuthorizedBotXMethod[str]): - """Method for sending smartapp notifications.""" - - __url__ = "/api/v3/botx/smartapps/notification" - __method__ = "POST" - __returning__ = str - - #: smartapp chat - group_chat_id: Optional[UUID] - - #: unread notifications count - smartapp_counter: int - - #: event options - opts: Dict[str, Any] = {} - - #: version of protocol smartapp <-> bot - smartapp_api_version: int diff --git a/botx/clients/methods/v3/stickers/__init__.py b/botx/clients/methods/v3/stickers/__init__.py deleted file mode 100644 index 1ec2ca1d..00000000 --- a/botx/clients/methods/v3/stickers/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for sticker resource.""" diff --git a/botx/clients/methods/v3/stickers/add_sticker.py b/botx/clients/methods/v3/stickers/add_sticker.py deleted file mode 100644 index 4891e3a6..00000000 --- a/botx/clients/methods/v3/stickers/add_sticker.py +++ /dev/null @@ -1,55 +0,0 @@ -"""Method for adding stickers into sticker pack.""" -from http import HTTPStatus -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import image_not_valid, sticker_pack_not_found -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import Sticker - - -class AddSticker(AuthorizedBotXMethod[Sticker]): - """Method for adding stickers into sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}/stickers/" - __method__ = "POST" - __returning__ = Sticker - __errors_handlers__ = { - HTTPStatus.BAD_REQUEST: ( - sticker_pack_not_found.handle_error, - image_not_valid.handle_error, - ), - } - - #: sticker pack ID. - pack_id: UUID - - #: emoji that the sticker will be associated with. - emoji: str - - #: sticker image. - image: str - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params={}, - json_body=dict(request_params), # type: ignore - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/create_sticker_pack.py b/botx/clients/methods/v3/stickers/create_sticker_pack.py deleted file mode 100644 index 78a20f5b..00000000 --- a/botx/clients/methods/v3/stickers/create_sticker_pack.py +++ /dev/null @@ -1,37 +0,0 @@ -"""Method for creating new sticker pack.""" -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import StickerPack - - -class CreateStickerPack(AuthorizedBotXMethod[StickerPack]): - """Method for creating sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs" - __method__ = "POST" - __returning__ = StickerPack - - #: sticker pack name. - name: str - - #: author HUID. - user_huid: UUID - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=dict(request_params), # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/delete_sticker.py b/botx/clients/methods/v3/stickers/delete_sticker.py deleted file mode 100644 index b6e89664..00000000 --- a/botx/clients/methods/v3/stickers/delete_sticker.py +++ /dev/null @@ -1,48 +0,0 @@ -"""Method for deleting sticker from sticker pack.""" -from http import HTTPStatus -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import sticker_pack_or_sticker_not_found -from botx.clients.types.http import HTTPRequest - - -class DeleteSticker(AuthorizedBotXMethod[str]): - """Method for deleting sticker from sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}/stickers/{sticker_id}" - __method__ = "DELETE" - __returning__ = str - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (sticker_pack_or_sticker_not_found.handle_error,), - } - - # : sticker pack ID. - pack_id: UUID - - # : sticker ID. - sticker_id: UUID - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id, sticker_id=self.sticker_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=request_params, # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/delete_sticker_pack.py b/botx/clients/methods/v3/stickers/delete_sticker_pack.py deleted file mode 100644 index 33f9a8b6..00000000 --- a/botx/clients/methods/v3/stickers/delete_sticker_pack.py +++ /dev/null @@ -1,45 +0,0 @@ -"""Method for deleting sticker pack.""" -from http import HTTPStatus -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import sticker_pack_not_found -from botx.clients.types.http import HTTPRequest - - -class DeleteStickerPack(AuthorizedBotXMethod[str]): - """Method for deleting sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}" - __method__ = "DELETE" - __returning__ = str - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (sticker_pack_not_found.handle_error,), - } - - # : sticker pack ID. - pack_id: UUID - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=request_params, # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/edit_sticker_pack.py b/botx/clients/methods/v3/stickers/edit_sticker_pack.py deleted file mode 100644 index c679750f..00000000 --- a/botx/clients/methods/v3/stickers/edit_sticker_pack.py +++ /dev/null @@ -1,56 +0,0 @@ -"""Method for editing sticker pack.""" -from http import HTTPStatus -from typing import List, Optional -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import sticker_pack_not_found -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import StickerPack - - -class EditStickerPack(AuthorizedBotXMethod[StickerPack]): - """Method for editing sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}" - __method__ = "PUT" - __returning__ = StickerPack - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (sticker_pack_not_found.handle_error,), - } - - # : sticker pack ID. - pack_id: UUID - - # : sticker pack name. - name: str - - #: sticker pack preview. - preview: Optional[UUID] - - #: stickers order in sticker pack. - stickers_order: Optional[List[UUID]] - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params={}, - json_body=request_params, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/sticker.py b/botx/clients/methods/v3/stickers/sticker.py deleted file mode 100644 index d50efe7d..00000000 --- a/botx/clients/methods/v3/stickers/sticker.py +++ /dev/null @@ -1,49 +0,0 @@ -"""Method for getting sticker from sticker pack.""" -from http import HTTPStatus -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import sticker_pack_or_sticker_not_found -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import StickerFromPack - - -class GetSticker(AuthorizedBotXMethod[StickerFromPack]): - """Method for getting sticker from sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}/stickers/{sticker_id}" - __method__ = "GET" - __returning__ = StickerFromPack - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (sticker_pack_or_sticker_not_found.handle_error,), - } - - #: sticker pack ID. - pack_id: UUID - - #: sticker ID. - sticker_id: UUID - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id, sticker_id=self.sticker_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=request_params, # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/sticker_pack.py b/botx/clients/methods/v3/stickers/sticker_pack.py deleted file mode 100644 index 03ee1c5f..00000000 --- a/botx/clients/methods/v3/stickers/sticker_pack.py +++ /dev/null @@ -1,46 +0,0 @@ -"""Method for getting sticker pack.""" -from http import HTTPStatus -from urllib.parse import urljoin -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors.stickers import sticker_pack_not_found -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import StickerPack - - -class GetStickerPack(AuthorizedBotXMethod[StickerPack]): - """Method for getting sticker pack.""" - - __url__ = "/api/v3/botx/stickers/packs/{pack_id}" - __method__ = "GET" - __returning__ = StickerPack - __errors_handlers__ = { - HTTPStatus.NOT_FOUND: (sticker_pack_not_found.handle_error,), - } - - #: sticker pack ID. - pack_id: UUID - - @property - def url(self) -> str: - """Full URL for request with filling pack_id.""" - api_url = self.__url__.format(pack_id=self.pack_id) - return urljoin(super().url, api_url) - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=request_params, # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/stickers/sticker_pack_list.py b/botx/clients/methods/v3/stickers/sticker_pack_list.py deleted file mode 100644 index 56c44999..00000000 --- a/botx/clients/methods/v3/stickers/sticker_pack_list.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Method for getting sticker pack list.""" -from typing import Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.types.http import HTTPRequest -from botx.models.stickers import StickerPackList - - -class GetStickerPackList(AuthorizedBotXMethod[StickerPackList]): - """Method for getting sticker pack list.""" - - __url__ = "/api/v3/botx/stickers/packs" - __method__ = "GET" - __returning__ = StickerPackList - - #: author HUID. - user_huid: Optional[UUID] - - #: returning value count. - limit: int - - #: cursor hash for pagination. - after: Optional[str] = None - - def build_http_request(self) -> HTTPRequest: - """Build HTTP request that can be used by clients for making real requests. - - Returns: - Built HTTP request. - """ - request_params = self.build_serialized_dict() - - return HTTPRequest.construct( - method=self.http_method, - url=self.url, - headers=self.headers, - query_params=dict(request_params), # type: ignore - json_body={}, - expected_type=self.expected_type, - ) diff --git a/botx/clients/methods/v3/users/__init__.py b/botx/clients/methods/v3/users/__init__.py deleted file mode 100644 index db4e74a6..00000000 --- a/botx/clients/methods/v3/users/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for users resource.""" diff --git a/botx/clients/methods/v3/users/by_email.py b/botx/clients/methods/v3/users/by_email.py deleted file mode 100644 index fed0c91b..00000000 --- a/botx/clients/methods/v3/users/by_email.py +++ /dev/null @@ -1,18 +0,0 @@ -"""Method for searching user by his email.""" -from http import HTTPStatus - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import user_not_found -from botx.models.users import UserFromSearch - - -class ByEmail(AuthorizedBotXMethod[UserFromSearch]): - """Method for searching user by his email.""" - - __url__ = "/api/v3/botx/users/by_email" - __method__ = "GET" - __returning__ = UserFromSearch - __errors_handlers__ = {HTTPStatus.NOT_FOUND: user_not_found.handle_error} - - #: email to search - email: str diff --git a/botx/clients/methods/v3/users/by_huid.py b/botx/clients/methods/v3/users/by_huid.py deleted file mode 100644 index 4cf3e404..00000000 --- a/botx/clients/methods/v3/users/by_huid.py +++ /dev/null @@ -1,19 +0,0 @@ -"""Method for searching user by his HUID.""" -from http import HTTPStatus -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import user_not_found -from botx.models.users import UserFromSearch - - -class ByHUID(AuthorizedBotXMethod[UserFromSearch]): - """Method for searching user by his HUID.""" - - __url__ = "/api/v3/botx/users/by_huid" - __method__ = "GET" - __returning__ = UserFromSearch - __errors_handlers__ = {HTTPStatus.NOT_FOUND: user_not_found.handle_error} - - #: HUID to search - user_huid: UUID diff --git a/botx/clients/methods/v3/users/by_login.py b/botx/clients/methods/v3/users/by_login.py deleted file mode 100644 index df05b731..00000000 --- a/botx/clients/methods/v3/users/by_login.py +++ /dev/null @@ -1,21 +0,0 @@ -"""Method for searching user by his AD credentials.""" -from http import HTTPStatus - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.errors import user_not_found -from botx.models.users import UserFromSearch - - -class ByLogin(AuthorizedBotXMethod[UserFromSearch]): - """Method for searching user by his AD credentials.""" - - __url__ = "/api/v3/botx/users/by_login" - __method__ = "GET" - __returning__ = UserFromSearch - __errors_handlers__ = {HTTPStatus.NOT_FOUND: user_not_found.handle_error} - - #: AD login to search - ad_login: str - - #: AD domain to search - ad_domain: str diff --git a/botx/clients/methods/v4/__init__.py b/botx/clients/methods/v4/__init__.py deleted file mode 100644 index 99f9d1af..00000000 --- a/botx/clients/methods/v4/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for V4 API methods for BotX API.""" diff --git a/botx/clients/methods/v4/notifications/__init__.py b/botx/clients/methods/v4/notifications/__init__.py deleted file mode 100644 index 1506ec03..00000000 --- a/botx/clients/methods/v4/notifications/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for methods for notifications resource.""" diff --git a/botx/clients/methods/v4/notifications/internal_bot_notification.py b/botx/clients/methods/v4/notifications/internal_bot_notification.py deleted file mode 100644 index 1b0d5144..00000000 --- a/botx/clients/methods/v4/notifications/internal_bot_notification.py +++ /dev/null @@ -1,29 +0,0 @@ -"""Method for sending internal bot notification.""" -from typing import Any, Dict, List, Optional -from uuid import UUID - -from botx.clients.methods.base import AuthorizedBotXMethod -from botx.clients.methods.extractors import extract_generated_sync_id -from botx.clients.types.message_payload import InternalBotNotificationPayload -from botx.clients.types.response_results import InternalBotNotificationResult - - -class InternalBotNotification(AuthorizedBotXMethod[UUID]): - """Method for sending internal bot notification.""" - - __url__ = "/api/v4/botx/notifications/internal" - __method__ = "POST" - __returning__ = InternalBotNotificationResult - __result_extractor__ = extract_generated_sync_id - - #: IDs of chats for new notification. - group_chat_id: UUID - - #: HUIDs of bots that should receive notifications (None for all bots in chat). - recipients: Optional[List[UUID]] = None - - # notification payload - data: InternalBotNotificationPayload # noqa: WPS110 - - #: extra options for message. - opts: Dict[str, Any] = {} diff --git a/botx/clients/types/__init__.py b/botx/clients/types/__init__.py deleted file mode 100644 index bdf2c69a..00000000 --- a/botx/clients/types/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Types that are used only for communicating with BotX API.""" diff --git a/botx/clients/types/http.py b/botx/clients/types/http.py deleted file mode 100644 index aae73d5d..00000000 --- a/botx/clients/types/http.py +++ /dev/null @@ -1,93 +0,0 @@ -"""Custom wrapper for HTTP request for BotX API.""" -from enum import Enum -from io import BytesIO -from typing import Any, Dict, List, Optional, Tuple, Union - -from pydantic import BaseModel, root_validator - -PrimitiveDataType = Union[None, str, int, float, bool] - - -class ExpectedType(Enum): - """Expected types of response body.""" - - JSON = "JSON" # noqa: WPS115 - BINARY = "BINARY" # noqa: WPS115 - - -class HTTPRequest(BaseModel): - """Wrapper for HTTP request.""" - - #: HTTP method. - method: str - - #: URL for request. - url: str - - #: headers for request. - headers: Dict[str, str] - - #: query params for request. - query_params: Dict[str, PrimitiveDataType] - - #: request body. - json_body: Optional[Dict[str, Any]] - - #: form data. - data: Optional[Dict[str, Any]] = None # noqa: WPS110 - - #: file for httpx in {field_name: (file_name, file_content)}. - files: Optional[Dict[str, Tuple[str, BytesIO]]] = None # noqa: WPS234 - - #: expected type of response body. - expected_type: ExpectedType = ExpectedType.JSON - - # This field is used to provide handlers that are not in the range of 400 to 599. - #: extra error codes. - should_process_as_error: List[int] = [] - - class Config: - arbitrary_types_allowed = True - - -class HTTPResponse(BaseModel): - """Wrapper for HTTP response.""" - - #: response headers - headers: Dict[str, str] - - #: response status code. - status_code: int - - #: response body. - json_body: Optional[Dict[str, Any]] = None - - #: response raw data. - raw_data: Optional[bytes] = None - - @property - def is_redirect(self) -> bool: - """Is redirect status code. - - Returns: - Check result. - """ - return 300 <= self.status_code < 399 # noqa: WPS432 - - @property - def is_error(self) -> bool: - """Is error status code. - - Returns: - Check result. - """ - return 400 <= self.status_code < 599 # noqa: WPS432 - - @root_validator(pre=True) - def check_fields(cls, values: Any) -> Any: # noqa: N805, WPS110 - """Check if passed both `json_body` and `raw_data`.""" - json_body, raw_data = values.get("json_body"), values.get("raw_data") - if (json_body is not None) and (raw_data is not None): - raise ValueError("you cannot pass both `json_body` and `raw_data`.") - - return values diff --git a/botx/clients/types/message_payload.py b/botx/clients/types/message_payload.py deleted file mode 100644 index dd4c8c25..00000000 --- a/botx/clients/types/message_payload.py +++ /dev/null @@ -1,72 +0,0 @@ -"""Shape that is used for messages from bot.""" -from typing import Any, Dict, List, Optional - -from pydantic import BaseModel, Field - -from botx.models.constants import MAXIMUM_TEXT_LENGTH -from botx.models.entities import Mention -from botx.models.enums import Statuses -from botx.models.messages.sending.options import ResultPayloadOptions -from botx.models.typing import BubbleMarkup, KeyboardMarkup - -try: - from typing import Literal # noqa: WPS433 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 - - -class ResultPayload(BaseModel): - """Data that is sent when bot answers on command or send notification.""" - - #: status of operation. - status: Literal[Statuses.ok] = Statuses.ok - - #: body for new message from bot. - body: str = Field("", max_length=MAXIMUM_TEXT_LENGTH) - - #: message metadata. - metadata: Dict[str, Any] = {} - - #: options for `notification` and `command_result` API entities. - opts: ResultPayloadOptions = ResultPayloadOptions() - - #: keyboard that will be used for new message. - keyboard: KeyboardMarkup = [] - - #: bubble elements that will be showed under new message. - bubble: BubbleMarkup = [] - - #: mentions that BotX API will append before new message text. - mentions: List[Mention] = [] - - -class UpdatePayload(BaseModel): - """Data that is sent when bot updates message.""" - - #: status of operation. - status: Literal[Statuses.ok] = Statuses.ok - - #: new body in message. - body: Optional[str] = Field(None, max_length=MAXIMUM_TEXT_LENGTH) - - #: message metadata. - metadata: Optional[Dict[str, Any]] = None - - #: new keyboard that will be used for new message. - keyboard: Optional[KeyboardMarkup] = None - - #: new bubble elements that will be showed under new message. - bubble: Optional[BubbleMarkup] = None - - #: new mentions that BotX API will append before new message text. - mentions: Optional[List[Mention]] = None - - -class InternalBotNotificationPayload(BaseModel): - """Data that is sent in internal bot notification.""" - - #: message data - message: str - - #: extra information about notification sender - sender: Optional[str] diff --git a/botx/clients/types/options.py b/botx/clients/types/options.py deleted file mode 100644 index 5059b12e..00000000 --- a/botx/clients/types/options.py +++ /dev/null @@ -1,17 +0,0 @@ -"""Special options for messages from bot.""" -from pydantic import BaseModel - -from botx.models.messages.sending.options import NotificationOptions - - -class ResultOptions(BaseModel): - """Configuration for command result or notification that is send to BotX API.""" - - #: send message only when stealth mode is enabled. - stealth_mode: bool = False - - #: use in-text mentions - raw_mentions: bool = False - - #: message options for configuring notifications. - notification_opts: NotificationOptions = NotificationOptions() diff --git a/botx/clients/types/response_results.py b/botx/clients/types/response_results.py deleted file mode 100644 index 506e830b..00000000 --- a/botx/clients/types/response_results.py +++ /dev/null @@ -1,25 +0,0 @@ -"""Responses from BotX API.""" -from uuid import UUID - -from pydantic import BaseModel - - -class PushResult(BaseModel): - """Entity that contains result from notification or command result push.""" - - #: event id of pushed message. - sync_id: UUID - - -class ChatCreatedResult(BaseModel): - """Entity that contains result from chat creation.""" - - #: id of created chat. - chat_id: UUID - - -class InternalBotNotificationResult(BaseModel): - """Entity that contains result from internal bot notification.""" - - #: event id of pushed message. - sync_id: UUID diff --git a/botx/clients/types/upload_file.py b/botx/clients/types/upload_file.py deleted file mode 100644 index 1847c422..00000000 --- a/botx/clients/types/upload_file.py +++ /dev/null @@ -1,14 +0,0 @@ -"""Uploading file metadata.""" -from typing import Optional - -from pydantic import BaseModel - - -class UploadingFileMeta(BaseModel): - """Uploading file metadata.""" - - #: duration of media file - duration: Optional[int] = None - - #: caption of media file - caption: Optional[str] = None diff --git a/botx/collecting/__init__.py b/botx/collecting/__init__.py deleted file mode 100644 index b20abee9..00000000 --- a/botx/collecting/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition of command handlers and routing mechanism.""" diff --git a/botx/collecting/collectors/__init__.py b/botx/collecting/collectors/__init__.py deleted file mode 100644 index fc68d408..00000000 --- a/botx/collecting/collectors/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for collecting logic.""" diff --git a/botx/collecting/collectors/base.py b/botx/collecting/collectors/base.py deleted file mode 100644 index 24784382..00000000 --- a/botx/collecting/collectors/base.py +++ /dev/null @@ -1,202 +0,0 @@ -"""Definition for base collector.""" - -from dataclasses import InitVar, field -from typing import Any, Callable, List, Optional, Sequence, Union - -from pydantic.dataclasses import dataclass - -from botx import converters -from botx.collecting.handlers.handler import Handler -from botx.collecting.handlers.name_generators import ( - get_body_from_name, - get_name_from_callable, -) -from botx.dependencies import models as deps -from botx.dependencies.models import Depends -from botx.exceptions import NoMatchFound - - -def _get_sorted_handlers(handlers: List[Handler]) -> List[Handler]: - return sorted(handlers, key=lambda handler: len(handler.body), reverse=True) - - -def _combine_dependencies( - *dependencies: Optional[Sequence[deps.Depends]], -) -> List[deps.Depends]: - result_dependencies = [] - for deps_sequence in dependencies: - result_dependencies.extend(converters.optional_sequence_to_list(deps_sequence)) - - return result_dependencies - - -def _check_new_handler_restrictions( - body: str, - name: Optional[str], - handler: Callable, - existed_handler: Handler, -) -> None: - handler_executor = existed_handler.handler - handler_name = existed_handler.name - - if body == existed_handler.body: - raise AssertionError("handler with body {0} already registered".format(body)) - - handler_registered = handler == handler_executor and name == handler_name - if name == existed_handler.name and not handler_registered: - raise AssertionError("handler with name {0} already registered".format(name)) - - -@dataclass -class BaseCollector: - """Base collector.""" - - default: InitVar[Handler] = None - - #: registered handlers on this collector handlers in order of adding. - handlers: List[Handler] = field(default_factory=list) - - #: handler that will be used for handling non matched message. - default_message_handler: Optional[Handler] = None - - #: background dependencies that will be executed for handlers. - dependencies: Optional[Sequence[Depends]] = None - - #: overrider for dependencies. - dependency_overrides_provider: Optional[Any] = None - - @property - def sorted_handlers(self) -> List[Handler]: - """Get added handlers sorted by bodies length. - - Returns: - Sorted handlers. - """ - return _get_sorted_handlers(self.handlers) - - def __post_init__(self, default: Optional[Handler]) -> None: - """Initialize or update special fields. - - Arguments: - default: callable that should be used as default handler. - """ - handlers = self.handlers - self.handlers = [] - self._add_handlers(converters.optional_sequence_to_list(handlers)) - - if default is not None: - self._add_default_handler(default) - - def handler_for(self, name: str) -> Handler: - """Find handler in handlers of this bot. - - Arguments: - name: name of handler that should be found. - - Returns: - Handler that was found by name. - - Raises: - NoMatchFound: raise if handler was not found. - """ - for handler in self.handlers: - if handler.name == name: - return handler - - raise NoMatchFound(search_param=name) - - def add_handler( # noqa: WPS211 - self, - handler: Callable, - *, - body: Optional[str] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[deps.Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> None: - """Create new handler from passed arguments and store it inside. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - body: body template that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - """ - if body is None: - name = name or get_name_from_callable(handler) - body = get_body_from_name(name) - - for registered_handler in self.handlers: - _check_new_handler_restrictions(body, name, handler, registered_handler) - - dep_override = ( - dependency_overrides_provider - if dependency_overrides_provider is not None - else self.dependency_overrides_provider - ) - command_handler = Handler( - body=body, - handler=handler, - name=name, # type: ignore - description=description, - full_description=full_description, # type: ignore - include_in_status=include_in_status, - dependencies=_combine_dependencies(self.dependencies, dependencies), - dependency_overrides_provider=dep_override, - ) - self.handlers.append(command_handler) - - def _add_handlers( - self, - handlers: List[Handler], - dependencies: Optional[Sequence[Depends]] = None, - ) -> None: - for handler in handlers: - combined_dependencies = _combine_dependencies( - dependencies, - handler.dependencies, - ) - self.add_handler( - body=handler.body, - handler=handler.handler, - name=handler.name, - description=handler.description, - full_description=handler.full_description, - include_in_status=handler.include_in_status, - dependencies=combined_dependencies, - ) - - def _add_default_handler( - self, - default: Handler, - dependencies: Optional[Sequence[Depends]] = None, - ) -> None: - default_dependencies = _combine_dependencies( - self.dependencies, - dependencies, - default.dependencies, - ) - self.default_message_handler = Handler( - body=default.body, - handler=default.handler, - name=default.name, - description=default.description, - full_description=default.full_description, - include_in_status=default.include_in_status, - dependencies=default_dependencies, - dependency_overrides_provider=self.dependency_overrides_provider, - ) diff --git a/botx/collecting/collectors/collector.py b/botx/collecting/collectors/collector.py deleted file mode 100644 index 07ab55db..00000000 --- a/botx/collecting/collectors/collector.py +++ /dev/null @@ -1,96 +0,0 @@ -"""Definition for collector.""" -from typing import Any, Optional, Sequence - -from loguru import logger -from pydantic.dataclasses import dataclass - -from botx.collecting.collectors.base import BaseCollector -from botx.collecting.collectors.mixins.default import DefaultHandlerMixin -from botx.collecting.collectors.mixins.handler import HandlerMixin -from botx.collecting.collectors.mixins.hidden import HiddenHandlerMixin -from botx.collecting.collectors.mixins.system_events import SystemEventsHandlerMixin -from botx.dependencies.models import Depends -from botx.exceptions import NoMatchFound -from botx.models.messages.message import Message - - -@dataclass -class Collector( # noqa: WPS215 - HandlerMixin, - DefaultHandlerMixin, - HiddenHandlerMixin, - SystemEventsHandlerMixin, - BaseCollector, -): - """Collector for different handlers.""" - - async def __call__(self, message: Message) -> None: - """Find handler and execute it. - - Arguments: - message: incoming message that will be passed to handler. - """ - await self.handle_message(message) - - def include_collector( - self, - collector: "Collector", - *, - dependencies: Optional[Sequence[Depends]] = None, - ) -> None: - """Include handlers from another collector into this one. - - Arguments: - collector: collector from which handlers should be copied. - dependencies: optional sequence of dependencies for handlers for this - collector. - - Raises: - AssertionError: raised if both collectors defines default handlers. - """ - if self.default_message_handler and collector.default_message_handler: - raise AssertionError("only one default handler can be applied") - - if collector.default_message_handler: - self._add_default_handler(collector.default_message_handler, dependencies) - - self._add_handlers(collector.handlers, dependencies) - - def command_for(self, *args: Any) -> str: - """Find handler and build a command string using passed body query_params. - - Arguments: - args: sequence of elements where first element should be name of handler. - - Returns: - Command string. - - Raises: - TypeError: raised no arguments passed. - """ - if not len(args): - raise TypeError("missing handler name as the first argument") - - return self.handler_for(args[0]).command_for(*args) - - async def handle_message(self, message: Message) -> None: - """Find handler and execute it. - - Arguments: - message: incoming message that will be passed to handler. - - Raises: - NoMatchFound: raised if no handler for message found. - """ - for handler in self.sorted_handlers: - if handler.matches(message): - logger.bind(botx_collector=True).info( - "botx => {0}: {1}".format(handler.name, message.command.command), - ) - await handler(message) - return - - if self.default_message_handler and not message.is_system_event: - await self.default_message_handler(message) - else: - raise NoMatchFound(search_param=message.body) diff --git a/botx/collecting/collectors/mixins/__init__.py b/botx/collecting/collectors/mixins/__init__.py deleted file mode 100644 index f175eefa..00000000 --- a/botx/collecting/collectors/mixins/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for mixin with decorators for collector.""" diff --git a/botx/collecting/collectors/mixins/default.py b/botx/collecting/collectors/mixins/default.py deleted file mode 100644 index cc37c988..00000000 --- a/botx/collecting/collectors/mixins/default.py +++ /dev/null @@ -1,102 +0,0 @@ -"""Definition for mixin with default decorator.""" -from functools import partial -from typing import Any, Callable, Optional, Sequence, Union, cast - -from botx.collecting.collectors.mixins.handler import HandlerDecoratorProtocol -from botx.collecting.handlers.handler import Handler -from botx.collecting.handlers.name_generators import get_name_from_callable -from botx.dependencies.models import Depends - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class HandlerSearchProtocol(Protocol): - """Protocol for searching handler.""" - - def handler_for(self, name: str) -> Handler: - """Find handler in handlers of this bot.""" - - -class DefaultHandlerMixin: - """Mixin that defines default handler decorator.""" - - default_message_handler: Optional[Handler] - - def default( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = False, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Add new handler to bot and register it as default handler. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - - Raises: - AssertionError: raised if default handler already defined on collector. - """ - if self.default_message_handler is not None: - raise AssertionError( - "default handler is already registered on this collector", - ) - - if handler: - registered_handler = cast(HandlerDecoratorProtocol, self).handler( - handler=handler, - command=command, - commands=commands, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - name = name or get_name_from_callable(registered_handler) - self.default_message_handler = cast( - HandlerSearchProtocol, - self, - ).handler_for(name) - - return handler - - return partial( - self.default, - command=command, - commands=commands, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/collecting/collectors/mixins/handler.py b/botx/collecting/collectors/mixins/handler.py deleted file mode 100644 index 15eadcf4..00000000 --- a/botx/collecting/collectors/mixins/handler.py +++ /dev/null @@ -1,124 +0,0 @@ -"""Definition for mixin with handler decorator.""" -from functools import partial -from typing import Any, Callable, List, Optional, Sequence, Union, cast - -from botx import converters -from botx.dependencies.models import Depends - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class AddHandlerProtocol(Protocol): - """Protocol for definition add_handler method.""" - - def add_handler( # noqa: WPS211 - self, - handler: Callable, - *, - body: Optional[str] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> None: - """Create new handler from passed arguments and store it inside.""" - - -class HandlerDecoratorProtocol(Protocol): - """Protocol for definition handler decorator.""" - - def handler( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Add new handler to collector.""" - - -class HandlerMixin: - """Mixin that defines handler decorator.""" - - def handler( # noqa: WPS211 - self: AddHandlerProtocol, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - description: Optional[str] = None, - full_description: Optional[str] = None, - include_in_status: Union[bool, Callable] = True, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Add new handler to collector. - - !!! info - If `include_in_status` is a function, then `body` argument will be checked - for matching public commands style, like `/command`. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - description: description for command that will be shown in bot's menu. - full_description: full description that can be used for example in `/help` - command. - include_in_status: should this handler be shown in bot's menu, can be - callable function with no arguments *(for now)*. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - if handler: - handler_commands: List[ - Optional[str] - ] = converters.optional_sequence_to_list(commands) - - if command and commands: - handler_commands.insert(0, command) - elif not commands: - handler_commands = [command] - - for command_body in handler_commands: - self.add_handler( - body=command_body, - handler=handler, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - return handler - - return partial( - cast(HandlerDecoratorProtocol, self).handler, - command=command, - commands=commands, - name=name, - description=description, - full_description=full_description, - include_in_status=include_in_status, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/collecting/collectors/mixins/hidden.py b/botx/collecting/collectors/mixins/hidden.py deleted file mode 100644 index 892ebacf..00000000 --- a/botx/collecting/collectors/mixins/hidden.py +++ /dev/null @@ -1,43 +0,0 @@ -"""Definition for mixin with hidden decorator.""" -from typing import Any, Callable, Optional, Sequence - -from botx.collecting.collectors.mixins.handler import HandlerDecoratorProtocol -from botx.dependencies.models import Depends - - -class HiddenHandlerMixin: - """Mixin that defines hidden handler decorator.""" - - def hidden( # noqa: WPS211 - self: HandlerDecoratorProtocol, - handler: Optional[Callable] = None, - *, - command: Optional[str] = None, - commands: Optional[Sequence[str]] = None, - name: Optional[str] = None, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register hidden handler that won't be showed in menu. - - Arguments: - handler: callable that will be used for executing handler. - command: body template that will trigger this handler. - commands: list of body templates that will trigger this handler. - name: optional name for handler that will be used in generating body. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.handler( - handler=handler, - command=command, - commands=commands, - name=name, - include_in_status=False, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/collecting/collectors/mixins/system_events.py b/botx/collecting/collectors/mixins/system_events.py deleted file mode 100644 index d8c79e33..00000000 --- a/botx/collecting/collectors/mixins/system_events.py +++ /dev/null @@ -1,289 +0,0 @@ -"""Definition for mixin with system events decorator.""" -from typing import Any, Callable, Optional, Sequence, cast - -from botx.collecting.collectors.mixins.handler import HandlerDecoratorProtocol -from botx.dependencies.models import Depends -from botx.models.enums import SystemEvents - -try: - from typing import Protocol # noqa: WPS433 -except ImportError: - from typing_extensions import Protocol # type: ignore # noqa: WPS433, WPS440, F401 - - -class SystemEventsHandlerMixin: # noqa: WPS214 - """Mixin that defines system events handler decorator.""" - - def system_event( # noqa: WPS211 - self, - handler: Optional[Callable] = None, - *, - event: Optional[SystemEvents] = None, - events: Optional[Sequence[SystemEvents]] = None, - name: Optional[str] = None, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for system event. - - Arguments: - handler: callable that will be used for executing handler. - event: event for triggering this handler. - events: a sequence of events that will trigger handler. - name: optional name for handler that will be used in generating body. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - - Raises: - AssertionError: raised if nor event or events passed. - """ - if not (event or events): - raise AssertionError("at least one event should be passed") - - return cast(HandlerDecoratorProtocol, self).handler( - handler=handler, - command=event.value if event else None, - commands=[event.value for event in events] if events else None, - name=name, - include_in_status=False, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def chat_created( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `system:chat_created` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.chat_created, - name=SystemEvents.chat_created.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def file_transfer( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `file_transfer` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.file_transfer, - name=SystemEvents.file_transfer.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def added_to_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `added_to_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.added_to_chat, - name=SystemEvents.added_to_chat.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def deleted_from_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `deleted_from_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.deleted_from_chat, - name=SystemEvents.deleted_from_chat.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def left_from_chat( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `left_from_chat` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.left_from_chat, - name=SystemEvents.left_from_chat.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def internal_bot_notification( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `internal_bot_notification` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.internal_bot_notification, - name=SystemEvents.internal_bot_notification.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def cts_login( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `cts_login` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.cts_login, - name=SystemEvents.cts_login.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def cts_logout( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `cts_logout` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.cts_logout, - name=SystemEvents.cts_logout.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) - - def smartapp_event( - self, - handler: Optional[Callable] = None, - *, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> Callable: - """Register handler for `smartapp_event` event. - - Arguments: - handler: callable that will be used for executing handler. - dependencies: sequence of dependencies that should be executed before - handler. - dependency_overrides_provider: mock of callable for handler. - - Returns: - Passed in `handler` callable. - """ - return self.system_event( - handler=handler, - event=SystemEvents.smartapp_event, - name=SystemEvents.smartapp_event.value, - dependencies=dependencies, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/collecting/handlers/__init__.py b/botx/collecting/handlers/__init__.py deleted file mode 100644 index 8b3be030..00000000 --- a/botx/collecting/handlers/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for handler related stuff.""" diff --git a/botx/collecting/handlers/handler.py b/botx/collecting/handlers/handler.py deleted file mode 100644 index db5ee1a0..00000000 --- a/botx/collecting/handlers/handler.py +++ /dev/null @@ -1,128 +0,0 @@ -"""Definition for command handler.""" - -from __future__ import annotations - -import re -from dataclasses import field -from typing import Any, Callable, List, Optional, Union - -from pydantic import validator -from pydantic.dataclasses import dataclass - -from botx.collecting.handlers.validators import ( - check_handler_is_function, - retrieve_dependant, - retrieve_executor, - retrieve_full_description_for_handler, - retrieve_name_for_handler, - validate_body_for_status, -) -from botx.dependencies import models as deps -from botx.models.messages.message import Message - - -@dataclass -class Handler: - """Handler that will store body and callable.""" - - #: callable for executing registered logic. - handler: Callable - - #: command body. - body: str - - #: name of handler. - name: str = "" - - #: description that will be shown in bot's menu. - description: Optional[str] = None - - #: custom description that can be used for another purposes. - full_description: str = "" - - #: should handler be included into status. - include_in_status: Union[bool, Callable] = True - - #: wrapper around handler that will be executed. - dependant: deps.Dependant = field(init=False) - - #: background dependencies for handler. - dependencies: List[deps.Depends] = field(default_factory=list) - - #: custom object that will override dependencies for handler. - dependency_overrides_provider: Optional[Any] = None - - #: function that will be used for handling incoming message - executor: Callable = field(init=False) - - _body_validator = validator("executor", always=True)(validate_body_for_status) - _handler_is_function_validator = validator("handler", pre=True, always=True)( - check_handler_is_function, - ) - - async def __call__(self, message: Message) -> None: - """Execute handler using incoming message. - - Arguments: - message: message that will be handled by handler. - """ - await self.executor(message) - - def __post_init__(self) -> None: - """Initialize or update special fields.""" - self.name = retrieve_name_for_handler(self.name, self.handler) - self.full_description = retrieve_full_description_for_handler( - self.full_description, - self.handler, - ) - self.dependant = retrieve_dependant(self.handler, self.dependencies) - self.executor = retrieve_executor( # type: ignore - self.dependant, - self.dependency_overrides_provider, - ) - - def matches(self, message: Message) -> bool: - """Check if message body matched to handler's body. - - Arguments: - message: incoming message which body will be used to check route. - - Returns: - Result of check. - """ - return bool(re.compile(self.body).match(message.body)) - - def command_for(self, *args: Any) -> str: - """Build a command string using passed body query_params. - - Arguments: - args: sequence of elements that are arguments for command. - - Returns: - Built command. - """ - args_str = " ".join((str(arg) for arg in args[1:])) - return "{0} {1}".format(self.body, args_str).strip() - - def __eq__(self, other: object) -> bool: - """Compare 2 handlers for equality. - - Arguments: - other: handler to compare with. - - Returns: - Result of comparing. - """ - if not isinstance(other, Handler): - return False - - callable_comp = self.handler == other.handler - callable_comp = callable_comp and self.dependencies == other.dependencies - - export_comp = self.name == other.name - export_comp = export_comp and self.body == other.body - export_comp = export_comp and self.description == other.description - export_comp = export_comp and self.full_description == other.full_description - export_comp = export_comp and self.include_in_status == other.include_in_status - - return callable_comp and export_comp diff --git a/botx/collecting/handlers/name_generators.py b/botx/collecting/handlers/name_generators.py deleted file mode 100644 index e2210ea9..00000000 --- a/botx/collecting/handlers/name_generators.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Generators for names for handler.""" - -import inspect -import re -from typing import Callable - - -def get_body_from_name(name: str) -> str: - """Get auto body from given handler name in format `/word-word`. - - Examples: - ``` - >>> get_body_from_name("HandlerFunction") - "handler-function" - >>> get_body_from_name("handlerFunction") - "handler-function" - ``` - Arguments: - name: name of handler for which body should be generated. - """ - splited_words = re.findall(r"^[a-z\d_\-]+|[A-Z\d_\-][^A-Z\d_\-]*", name) - joined_body = "-".join(splited_words) - dashed_body = joined_body.replace("_", "-") - return "/{0}".format(re.sub("-+", "-", dashed_body).lower()) - - -def get_name_from_callable(handler: Callable) -> str: - """Get auto name from given callable object. - - Arguments: - handler: callable object that will be used to retrieve auto name for handler. - - Returns: - Name obtained from callable. - """ - is_function = inspect.isfunction(handler) - is_method = inspect.ismethod(handler) - is_class = inspect.isclass(handler) - if is_function or is_method or is_class: - return handler.__name__ - return handler.__class__.__name__ diff --git a/botx/collecting/handlers/validators.py b/botx/collecting/handlers/validators.py deleted file mode 100644 index 52cbc405..00000000 --- a/botx/collecting/handlers/validators.py +++ /dev/null @@ -1,127 +0,0 @@ -"""Validators and extractors for Handler fields.""" -import inspect -from typing import Any, Callable, List, Optional - -from botx.collecting.handlers.name_generators import get_name_from_callable -from botx.dependencies.models import Dependant, Depends, get_dependant -from botx.dependencies.solving import get_executor - - -def validate_body_for_status(executor: Callable, values: dict) -> Callable: - """Validate that body is acceptable for status. - - Arguments: - executor: executor that will be just returned from validator. - values: already checked validated_values. - - Returns: - Passed executor. - - Raises: - ValueError: raised if body is not acceptable for status. - """ - include_in_status = values["include_in_status"] - if not include_in_status: - return executor - - body = values["body"] - - if not body.startswith("/"): - raise ValueError("public commands should start with leading slash") - - slash_part_of_body = body[: -len(body.strip("/"))] - if slash_part_of_body.count("/") != 1: - raise ValueError("command body can contain only single leading slash") - - if len(body.split()) != 1: - raise ValueError("public commands should contain only one word") - - return executor - - -def retrieve_name_for_handler(name: Optional[str], handler: Callable) -> str: - """Retrieve name for handler. - - Arguments: - name: passed name for handler. - handler: handler that will be used to generate name. - - Returns: - Name for handler. - """ - return name or get_name_from_callable(handler) - - -def retrieve_full_description_for_handler( - full_description: Optional[str], - handler: Callable, -) -> str: - """Retrieve full description for handler. - - Arguments: - full_description: passed name for handler. - handler: handler from which docstring will be used. - - Returns: - Full description for handler. - """ - return full_description or inspect.cleandoc(handler.__doc__ or "") - - -def check_handler_is_function(handler: Callable) -> Callable: - """Check handler can be proceed by library. - - Library can handle functions and methods as handlers. - - Arguments: - handler: passed handler callable. - - Returns: - Passed handler. - - Raises: - ValueError: raised if handler is not acceptable. - """ - if not (inspect.isfunction(handler) or inspect.ismethod(handler)): - raise ValueError("handler must be a function or method") - - return handler - - -def retrieve_dependant(handler: Callable, dependencies: List[Depends]) -> Dependant: - """Retrieve dependant for handler. - - Arguments: - handler: handler for which dependant should be created. - dependencies: passed background dependencies. - - Returns: - Generated dependant object. - """ - dependant = get_dependant(call=handler) - for index, depends in enumerate(dependencies): - dependant.dependencies.insert( - index, - get_dependant(call=depends.dependency, use_cache=depends.use_cache), - ) - - return dependant - - -def retrieve_executor( - dependant: Dependant, - dependency_overrides_provider: Any, -) -> Callable: - """Retrieve executor for handler. - - Arguments: - dependant: dependant that will be used to generate executor. - dependency_overrides_provider: overrider for dependencies. - - Returns: - Generated executor. - """ - return get_executor( - dependant=dependant, - dependency_overrides_provider=dependency_overrides_provider, - ) diff --git a/botx/concurrency.py b/botx/concurrency.py deleted file mode 100644 index 9e15ab3b..00000000 --- a/botx/concurrency.py +++ /dev/null @@ -1,104 +0,0 @@ -"""Helpers for execution functions as coroutines.""" - -import asyncio -import contextvars -import functools -import inspect -from typing import Any, Callable, Coroutine - - -def is_awaitable_object(call: Callable) -> bool: - """Check if object is an awaitable or an object which __call__ method is awaitable. - - Arguments: - call: callable for checking. - - Returns: - Result of check. - """ - if is_awaitable(call): - return True - call = getattr(call, "__call__", None) # noqa: B004 - return asyncio.iscoroutinefunction(call) - - -def is_awaitable(call: Callable) -> bool: - """Check if function returns awaitable object. - - Arguments: - call: function that should be checked. - - Returns: - Result of check. - """ - if inspect.isfunction(call) or inspect.ismethod(call): - return asyncio.iscoroutinefunction(call) - return False - - -async def run_in_threadpool(call: Callable, *args: Any, **kwargs: Any) -> Any: - """Run regular function (not a coroutine) as awaitable coroutine. - - Arguments: - call: function that should be called as coroutine. - args: positional arguments for the function. - kwargs: keyword arguments for the function. - - Returns: - Result of function call. - """ - loop = asyncio.get_event_loop() - child = functools.partial(call, *args, **kwargs) - context = contextvars.copy_context() - call = context.run - args = (child,) - return await loop.run_in_executor(None, call, *args) - - -def callable_to_coroutine(func: Callable, *args: Any, **kwargs: Any) -> Coroutine: - """Transform callable to coroutine. - - Arguments: - func: function that can be sync or async and should be transformed into - coroutine. - args: positional arguments for this function. - kwargs: key arguments for this function. - - Returns: - Coroutine object from passed callable. - """ - if is_awaitable_object(func): - return func(*args, **kwargs) - - return run_in_threadpool(func, *args, **kwargs) - - -def run_in_blocking_loop(call: Callable, *args: Any, **kwargs: Any) -> Any: - """Run coroutine with loop blocking. - - Arguments: - call: function that should be called in loop until complete. - args: function arguments. - kwargs: function key arguments. - - Returns: - Result of function call. - """ - try: - loop = asyncio.get_event_loop() - except RuntimeError: - loop = asyncio.new_event_loop() - - return loop.run_until_complete(callable_to_coroutine(call, *args, **kwargs)) - - -def async_to_sync(func: Callable) -> Callable: - """Convert asynchronous function to blocking. - - Arguments: - func: function that should be converted. - - Returns: - Converted function. - """ - return functools.partial(run_in_blocking_loop, func) diff --git a/botx/constants.py b/botx/constants.py new file mode 100644 index 00000000..085fb2d0 --- /dev/null +++ b/botx/constants.py @@ -0,0 +1,12 @@ +try: + from typing import Final +except ImportError: + from typing_extensions import Final # type: ignore # noqa: WPS440 + +CHUNK_SIZE: Final = 1024 * 1024 # 1Mb +BOT_API_VERSION: Final = 4 +SMARTAPP_API_VERSION: Final = 1 +STICKER_IMAGE_MAX_SIZE: Final = 512 * 1024 # 512Kb +STICKER_PACKS_PER_PAGE: Final = 10 +MAX_NOTIFICATION_BODY_LENGTH: Final = 4096 +MAX_FILE_LEN_IN_LOGS: Final = 64 diff --git a/botx/converters.py b/botx/converters.py index bb41049d..3a9147a9 100644 --- a/botx/converters.py +++ b/botx/converters.py @@ -1,19 +1,9 @@ -"""Converters for common operations.""" - from typing import List, Optional, Sequence, TypeVar -TSequenceElement = TypeVar("TSequenceElement") +TItem = TypeVar("TItem") def optional_sequence_to_list( - seq: Optional[Sequence[TSequenceElement]] = None, -) -> List[TSequenceElement]: - """Convert optional sequence of elements to list. - - Arguments: - seq: sequence that should be converted to list. - - Returns: - List of passed elements. - """ - return list(seq or []) + optional_sequence: Optional[Sequence[TItem]], +) -> List[TItem]: + return list(optional_sequence or []) diff --git a/botx/dependencies/__init__.py b/botx/dependencies/__init__.py deleted file mode 100644 index 5578cd4a..00000000 --- a/botx/dependencies/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Dependency injection implementation.""" diff --git a/botx/dependencies/injection_params.py b/botx/dependencies/injection_params.py deleted file mode 100644 index 5f7f0a47..00000000 --- a/botx/dependencies/injection_params.py +++ /dev/null @@ -1,19 +0,0 @@ -"""Wrappers around param classes that are used in handlers or dependencies.""" - -from typing import Any, Callable - -from botx.dependencies import models - - -def Depends(dependency: Callable, *, use_cache: bool = True) -> Any: # noqa: N802 - """Wrap Depends param for using in handlers. - - Arguments: - dependency: callable object that will be used in handlers or other dependencies - instances. - use_cache: use cache for dependency. - - Returns: - [Depends][botx.dependencies.models.Depends] that wraps passed callable. - """ - return models.Depends(dependency=dependency, use_cache=use_cache) diff --git a/botx/dependencies/inspecting.py b/botx/dependencies/inspecting.py deleted file mode 100644 index d14fc426..00000000 --- a/botx/dependencies/inspecting.py +++ /dev/null @@ -1,51 +0,0 @@ -"""Functions for inspecting signatures and parameters.""" - -import inspect -from typing import Any, Callable, Dict - -from pydantic.typing import ForwardRef, evaluate_forwardref - - -def get_typed_signature(call: Callable) -> inspect.Signature: - """Get signature for callable function with solving possible annotations. - - Arguments: - call: callable object that will be used to get signature with annotations. - - Returns: - Callable signature obtained. - """ - signature = inspect.signature(call) - global_namespace = getattr(call, "__globals__", {}) - typed_params = [ - inspect.Parameter( - name=dependency_param.name, - kind=dependency_param.kind, - default=dependency_param.default, - annotation=get_typed_annotation(dependency_param, global_namespace), - ) - for dependency_param in signature.parameters.values() - ] - return inspect.Signature(typed_params) - - -def get_typed_annotation( - dependency_param: inspect.Parameter, - global_namespace: Dict[str, Any], -) -> Any: - """Solve forward reference annotation for instance of `inspect.Parameter`. - - Arguments: - dependency_param: instance of `inspect.Parameter` for which possible forward - annotation will be evaluated. - global_namespace: dictionary of entities that can be used for evaluating - forward references. - - Returns: - Parameter annotation. - """ - annotation = dependency_param.annotation - if isinstance(annotation, str): - annotation = ForwardRef(annotation) - annotation = evaluate_forwardref(annotation, global_namespace, global_namespace) - return annotation diff --git a/botx/dependencies/models.py b/botx/dependencies/models.py deleted file mode 100644 index 36f61b1e..00000000 --- a/botx/dependencies/models.py +++ /dev/null @@ -1,167 +0,0 @@ -"""Dependant model and transforming functions.""" -from __future__ import annotations - -import inspect -from dataclasses import field -from typing import Any, Callable, Dict, List, Optional - -from pydantic.dataclasses import dataclass -from pydantic.utils import lenient_issubclass - -from botx.bots import bots -from botx.clients.clients import async_client, sync_client -from botx.dependencies import inspecting -from botx.models.messages.message import Message - -WRONG_PARAM_TYPE_ERROR_TEXT = ( - "Param {0} of {1} can only be a dependency, message, bot or client, got: {2}" -) - -CacheKey = Optional[Callable] -DependenciesCache = Dict[CacheKey, Any] - - -@dataclass -class Depends: - """Stores dependency callable.""" - - #: callable object that will be used in handlers or other dependencies instances. - dependency: Callable[..., Any] - - #: use cache for dependency. - use_cache: bool = True - - -@dataclass -class Dependant: - """Main model that contains all necessary data for solving dependencies.""" - - #: list of sub-dependencies for this dependency. - dependencies: List[Dependant] = field(default_factory=list) - - #: name of dependency. - name: Optional[str] = None - - #: callable object that will solve dependency. - call: Optional[Callable] = None - - #: param name for passing incoming [message][botx.models.messages.Message] - message_param_name: Optional[str] = None - - #: param name for passing [bot][botx.bots.Bot] that handles command. - bot_param_name: Optional[str] = None - - #: param name for passing [client][botx.clients.clients.async_client.AsyncClient]. - async_client_param_name: Optional[str] = None - - #: param name for passing [client][botx.clients.clients.sync_client.Client]. - sync_client_param_name: Optional[str] = None - - #: use cache for optimize solving performance. - use_cache: bool = True - - # Save the cache key at creation to optimize performance - #: storage for cache. - cache_key: CacheKey = field(init=False) - - def __post_init__(self) -> None: - """Init special fields.""" - self.cache_key = self.call - - -Dependant.__pydantic_model__.update_forward_refs() # type: ignore # noqa: WPS609 - - -def get_param_sub_dependant(*, dependency_param: inspect.Parameter) -> Dependant: - """Parse instance of parameter to get it as dependency. - - Arguments: - dependency_param: param for which sub dependency should be retrieved. - - Returns: - Object that will be used in solving dependency. - """ - depends: Depends = dependency_param.default - dependency = depends.dependency - - return get_dependant( - call=dependency, - name=dependency_param.name, - use_cache=depends.use_cache, - ) - - -def get_dependant( - *, - call: Callable, - name: Optional[str] = None, - use_cache: bool = True, -) -> Dependant: - """Get dependant instance from passed callable object. - - Arguments: - call: callable object that will be parsed to get required parameters and - sub dependencies. - name: name for dependency. - use_cache: use cache for optimize solving performance. - - Returns: - Object that will be used in solving dependency. - - Raises: - ValueError: raised if param is not Dependant or special type. - """ - dependant = Dependant(call=call, name=name, use_cache=use_cache) - for dependency_param in inspecting.get_typed_signature(call).parameters.values(): - if isinstance(dependency_param.default, Depends): - dependant.dependencies.append( - get_param_sub_dependant(dependency_param=dependency_param), - ) - continue - - is_special_param = add_special_param_to_dependency( - dependency_param=dependency_param, - dependant=dependant, - ) - if is_special_param: - continue - - raise ValueError( - WRONG_PARAM_TYPE_ERROR_TEXT.format( - dependency_param.name, - call, - dependency_param.annotation, - ), - ) - - return dependant - - -def add_special_param_to_dependency( - *, - dependency_param: inspect.Parameter, - dependant: Dependant, -) -> bool: - """Check if param is non field object that should be passed into callable. - - Arguments: - dependency_param: param that should be checked. - dependant: dependency which field would be filled with required param name. - - Returns: - Result of check. - """ - if lenient_issubclass(dependency_param.annotation, bots.Bot): - dependant.bot_param_name = dependency_param.name - return True - elif lenient_issubclass(dependency_param.annotation, Message): - dependant.message_param_name = dependency_param.name - return True - elif lenient_issubclass(dependency_param.annotation, async_client.AsyncClient): - dependant.async_client_param_name = dependency_param.name - return True - elif lenient_issubclass(dependency_param.annotation, sync_client.Client): - dependant.sync_client_param_name = dependency_param.name - return True - - return False diff --git a/botx/dependencies/solving.py b/botx/dependencies/solving.py deleted file mode 100644 index 2cda9de1..00000000 --- a/botx/dependencies/solving.py +++ /dev/null @@ -1,145 +0,0 @@ -"""Functions for solving dependencies.""" - -from typing import Any, Awaitable, Callable, Dict, Optional, Tuple, cast - -from botx import concurrency -from botx.dependencies.models import ( - CacheKey, - Dependant, - DependenciesCache, - get_dependant, -) -from botx.models.messages.message import Message - - -async def solve_sub_dependency( - message: Message, - dependant: Dependant, - solved_values: Dict[str, Any], - dependency_overrides_provider: Any, - dependency_cache: Dict[CacheKey, Any], -) -> None: - """ - Solve single sub dependency. - - Arguments: - message: incoming message that is used for solving this sub dependency. - dependant: dependency that is solving while calling this function. - solved_values: already filled validated_values that are required for this - dependency. - dependency_overrides_provider: an object with `dependency_overrides` attribute - that contains overrides for dependencies. - dependency_cache: cache that contains already solved dependency and result for - it. - - """ - call = cast(Callable, dependant.call) - use_sub_dependant = dependant - - overrides = getattr( - dependency_overrides_provider - if dependency_overrides_provider is not None - else message.bot, - "dependency_overrides", - {}, - ) - if overrides: - call = overrides.get(dependant.call, dependant.call) - use_sub_dependant = get_dependant(call=call, name=dependant.name) - - solving_result = await solve_dependencies( - message=message, - dependant=use_sub_dependant, - dependency_overrides_provider=dependency_overrides_provider, - dependency_cache=dependency_cache, - ) - dependency_cache.update(solving_result[1]) - - dependant.cache_key = dependant.cache_key - if dependant.use_cache and dependant.cache_key in dependency_cache: - solved = dependency_cache[dependant.cache_key] - else: - solved = await concurrency.callable_to_coroutine(call, **solving_result[0]) - - if dependant.name is not None: - solved_values[dependant.name] = solved - if dependant.cache_key not in dependency_cache: - dependency_cache[dependant.cache_key] = solved - - -async def solve_dependencies( - *, - message: Message, - dependant: Dependant, - dependency_overrides_provider: Any = None, - dependency_cache: Optional[Dict[CacheKey, Any]] = None, -) -> Tuple[Dict[str, Any], DependenciesCache]: - """ - Resolve all required dependencies for Dependant using incoming message. - - Arguments: - message: incoming Message with all necessary data. - dependant: Dependant object for which all sub dependencies should be solved. - dependency_overrides_provider: an object with `dependency_overrides` attribute - that contains overrides for dependencies. - dependency_cache: cache that contains already solved dependency and result for - it. - - Returns: - Keyword arguments with their vales and cache. - - """ - solved_values: Dict[str, Any] = {} - dependency_cache = dependency_cache or {} - for sub_dependant in dependant.dependencies: - await solve_sub_dependency( - message=message, - dependant=sub_dependant, - solved_values=solved_values, - dependency_overrides_provider=dependency_overrides_provider, - dependency_cache=dependency_cache, - ) - - if dependant.message_param_name: - solved_values[dependant.message_param_name] = message - if dependant.bot_param_name: - solved_values[dependant.bot_param_name] = message.bot - if dependant.async_client_param_name: - solved_values[dependant.async_client_param_name] = message.bot.client - if dependant.sync_client_param_name: - solved_values[dependant.sync_client_param_name] = message.bot.sync_client - return solved_values, dependency_cache - - -def get_executor( - dependant: Dependant, - dependency_overrides_provider: Any = None, -) -> Callable[[Message], Awaitable[None]]: - """Get an execution callable for passed dependency. - - Arguments: - dependant: passed dependency for which execution callable should be generated. - dependency_overrides_provider: dependency overrider that will be passed to the - execution. - - Returns: - Asynchronous executor for handling message. - - Raises: - AssertionError: raised if there is no callable in `dependant.call`. - """ - if dependant.call is None: - raise AssertionError("dependant.call must be present") - - async def factory(message: Message) -> None: - solved_values, _ = await solve_dependencies( - message=message, - dependant=dependant, - dependency_overrides_provider=dependency_overrides_provider, - ) - await concurrency.callable_to_coroutine( - cast(Callable, dependant.call), - **solved_values, - ) - - return factory diff --git a/botx/exception_handlers.py b/botx/exception_handlers.py deleted file mode 100644 index 9262bc35..00000000 --- a/botx/exception_handlers.py +++ /dev/null @@ -1,29 +0,0 @@ -"""Define several handlers for builtin exceptions from this library.""" - -from typing import Any - -from loguru import logger - -from botx.exceptions import NoMatchFound -from botx.models.messages import message as messages - - -async def dependency_failure_exception_handler(*_: Any) -> None: - """Just do nothing if there is this error, since it's just a signal for stop. - - Arguments: - _: default arguments passed to exception handler. - """ - - -async def no_match_found_exception_handler( - exception: NoMatchFound, - message: messages.Message, -) -> None: - """Log that handler was not found. - - Arguments: - exception: raised NoMatchFound exception. - message: message on which processing error was raised. - """ - logger.info("handler for {0!r} was not found", exception.search_param) diff --git a/botx/exceptions.py b/botx/exceptions.py deleted file mode 100644 index de36642b..00000000 --- a/botx/exceptions.py +++ /dev/null @@ -1,112 +0,0 @@ -"""Exceptions that are used in this library.""" -from typing import Any, Dict -from uuid import UUID - - -class BotXException(Exception): - """Base error for exception in this library.""" - - #: template that should be rendered on __str__ call. - message_template: str = "" - - def __init__(self, **kwargs: Any) -> None: - """Init exception with passed query_params. - - Arguments: - kwargs: key-arguments that will be stored in instance. - """ - self.__dict__ = kwargs - - def __str__(self) -> str: - """Render string representation. - - Returns: - String representation of error. - """ - return self.message_template.format(**self.__dict__) - - -class NoMatchFound(BotXException): - """Raised by collector if no matching handler exists.""" - - message_template = "handler for {search_param} not found" - - #: body for which handler was not found. - search_param: str - - -class DependencyFailure(BotXException): - """Raised when there is error in dependency and flow should be stopped.""" - - -class BotXAPIError(BotXException): - """Raised if there is an error in requests to BotX API.""" - - message_template = ( - "unable to send {method} {url} to BotX API ({status}): {response_content}" - ) - - #: URL from request. - url: str - - #: HTTP method. - method: str - - #: response from API. - response_content: Dict[str, Any] - - # HTTP status code. - status: int - - -class UnknownBotError(BotXException): - """Raised if bot does not know bot.""" - - message_template = "unknown bot {bot_id}" - - #: bot id that is unregistered. - bot_id: UUID - - -class BotXAPIRouteDeprecated(BotXAPIError): - """Raised if API route was deprecated.""" - - message_template = "route {method} {url} is deprecated" - - #: bot id that is unregistered. - bot_id: UUID - - -class TokenError(BotXException): - """Raised if token is invalid.""" - - message_template = "invalid token for bot {bot_id}" - - bot_id: UUID - - -class BotXJSONDecodeError(BotXException): - """Raised if response body cannot be processed.""" - - message_template = "unable to process response body from {method} {url}" - - #: URL from request. - url: str - - #: HTTP method. - method: str - - -class BotXConnectError(BotXException): - """Raised if unable to connect to service.""" - - message_template = ( - "unable to connect to service {method} {url}. " - "Make sure you specified the correct host in bot credentials." - ) - - #: URL from request. - url: str - - #: HTTP method. - method: str diff --git a/botx/image_validators.py b/botx/image_validators.py new file mode 100644 index 00000000..d66e30c3 --- /dev/null +++ b/botx/image_validators.py @@ -0,0 +1,23 @@ +from botx.async_buffer import AsyncBufferReadable, get_file_size +from botx.constants import STICKER_IMAGE_MAX_SIZE + +PNG_MAGIC_BYTES: bytes = b"\x89\x50\x4E\x47\x0D\x0A\x1A\x0A" + + +async def ensure_file_content_is_png(async_buffer: AsyncBufferReadable) -> None: + magic_bytes = await async_buffer.read(8) + + await async_buffer.seek(0) + + if magic_bytes != PNG_MAGIC_BYTES: + raise ValueError("Passed file is not PNG") + + +async def ensure_sticker_image_size_valid(async_buffer: AsyncBufferReadable) -> None: + file_size = await get_file_size(async_buffer) + + if file_size > STICKER_IMAGE_MAX_SIZE: + max_file_size_mb = STICKER_IMAGE_MAX_SIZE / 1024 / 1024 + raise ValueError( + f"Passed file size is greater than {max_file_size_mb:.1f} Mb", + ) diff --git a/botx/logger.py b/botx/logger.py new file mode 100644 index 00000000..ef97cab6 --- /dev/null +++ b/botx/logger.py @@ -0,0 +1,48 @@ +import json +from copy import deepcopy +from typing import TYPE_CHECKING, Any, Dict + +from loguru import logger as _logger + +from botx.constants import MAX_FILE_LEN_IN_LOGS + +if TYPE_CHECKING: # To avoid circular import + from loguru import Logger + + +def pformat_jsonable_obj(jsonable_obj: Any) -> str: + return json.dumps(jsonable_obj, sort_keys=True, indent=4, ensure_ascii=False) + + +def trim_file_data_in_outgoing_json(json_body: Any) -> Any: + if not isinstance(json_body, dict): + return json_body + + if json_body.get("file"): + json_body = deepcopy(json_body) + json_body["file"]["data"] = ( + json_body["file"]["data"][:MAX_FILE_LEN_IN_LOGS] + "..." + ) + + return json_body + + +def trim_file_data_in_incoming_json(json_body: Dict[str, Any]) -> Dict[str, Any]: + if json_body.get("attachments"): + # Max one attach per-message + # Link and Location doesn't have content + if json_body["attachments"][0]["data"].get("content"): + json_body = deepcopy(json_body) + json_body["attachments"][0]["data"]["content"] = ( + json_body["attachments"][0]["data"]["content"][:MAX_FILE_LEN_IN_LOGS] + + "..." + ) + + return json_body + + +def setup_logger() -> "Logger": + return _logger + + +logger = setup_logger() diff --git a/botx/middlewares/__init__.py b/botx/middlewares/__init__.py deleted file mode 100644 index 42632b0a..00000000 --- a/botx/middlewares/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition ob built-in middlewares for botx.""" diff --git a/botx/middlewares/authorization.py b/botx/middlewares/authorization.py deleted file mode 100644 index e0abc69b..00000000 --- a/botx/middlewares/authorization.py +++ /dev/null @@ -1,27 +0,0 @@ -"""Middleware for retrieving tokens from BotX API before processing message.""" - -from botx.middlewares.base import BaseMiddleware -from botx.models.messages.message import Message -from botx.typing import AsyncExecutor - - -class AuthorizationMiddleware(BaseMiddleware): - """Middleware for retrieving tokens from BotX API before processing message.""" - - async def dispatch(self, message: Message, call_next: AsyncExecutor) -> None: - """Obtain token for bot for handling answers to message. - - Arguments: - message: incoming message. - call_next: next executor in chain. - """ - bot = message.bot - bot_account = bot.get_account_by_bot_id(message.bot_id) - if bot_account.token is None: - token = await bot.get_token( - message.host, - message.bot_id, - bot_account.signature, - ) - bot_account.token = token - await call_next(message) diff --git a/botx/middlewares/base.py b/botx/middlewares/base.py deleted file mode 100644 index 312e6edf..00000000 --- a/botx/middlewares/base.py +++ /dev/null @@ -1,65 +0,0 @@ -"""Definition of base for custom middlewares. - -Important: - Middleware should implement `dispatch` method that can be a common function or - an asynchronous function. - -```python -class MyAsyncBotXMiddleware(BaseMiddleware): - async def dispatch( - self, message: Message, call_next: AsyncExecutor, - ) -> None: - await call_next(message) - -class MySyncBotXMiddleware(BaseMiddleware): - def dispatch(self, message: Message, call_next: SyncExecutor) -> None: - call_next(message) -``` -""" - -from typing import Callable, Optional - -from botx import concurrency -from botx.models.messages.message import Message -from botx.typing import Executor, MiddlewareDispatcher, SyncExecutor - - -def _default_dispatch( - _middleware: "BaseMiddleware", - _message: Message, - _call_next: SyncExecutor, -) -> None: - raise NotImplementedError - - -class BaseMiddleware: - """Base middleware entity.""" - - dispatch: Callable = _default_dispatch - - def __init__( - self, - executor: Executor, - dispatch: Optional[MiddlewareDispatcher] = None, - ) -> None: - """Init middleware with required query_params. - - Arguments: - executor: callable object that accept message and will be executed after - middlewares. - dispatch: middleware logic executor. - """ - self.executor = executor - self.dispatch_func = dispatch or self.dispatch - - async def __call__(self, message: Message) -> None: - """Call middleware dispatcher as normal handler executor. - - Arguments: - message: incoming message. - """ - executor = self.executor - if not concurrency.is_awaitable(self.dispatch_func): - executor = concurrency.async_to_sync(self.executor) - - await concurrency.callable_to_coroutine(self.dispatch_func, message, executor) diff --git a/botx/middlewares/exceptions.py b/botx/middlewares/exceptions.py deleted file mode 100644 index 54f2187c..00000000 --- a/botx/middlewares/exceptions.py +++ /dev/null @@ -1,133 +0,0 @@ -"""Definition of base middleware class and some default middlewares.""" - -from typing import Callable, Dict, Optional, Type - -from loguru import logger - -from botx import concurrency -from botx.middlewares.base import BaseMiddleware -from botx.models import files -from botx.models.messages.message import Message -from botx.typing import AsyncExecutor, Executor - - -class ExceptionMiddleware(BaseMiddleware): - """Custom middleware that is default and used to handle registered errors.""" - - def __init__(self, executor: Executor) -> None: - """Init middleware with required query_params. - - Arguments: - executor: callable object that accept message and will be executed after - middleware. - """ - super().__init__(executor) - self._exception_handlers: Dict[Type[Exception], Callable] = {} - - async def dispatch(self, message: Message, call_next: AsyncExecutor) -> None: - """Wrap executor for catching exception or log them. - - Arguments: - message: incoming message that will be passed to executor. - call_next: next executor that should be called after this. - """ - try: - await call_next(message) - except Exception as exc: - await self._handle_error_in_handler(exc, message) - - def add_exception_handler( - self, - exc_class: Type[Exception], - handler: Callable, - ) -> None: - """Register handler for specific exception in middleware. - - Arguments: - exc_class: exception class that should be handled by middleware. - handler: handler for exception. - """ - self._exception_handlers[exc_class] = handler - - def _lookup_handler_for_exception(self, exc: Exception) -> Optional[Callable]: - """Find handler for exception. - - Arguments: - exc: catched exception for which handler should be found. - - Returns: - Found handler or None. - """ - for exc_cls in type(exc).mro(): - handler = self._exception_handlers.get(exc_cls) - if handler: - return handler - - return None - - async def _handle_error_in_handler(self, exc: Exception, message: Message) -> None: - """Pass error back to handler if there is one or log error. - - Arguments: - exc: exception that occurred. - message: message on which exception occurred. - """ - exception_logger = logger.bind( - botx_error=True, - payload=message.incoming_message.copy( - update={ - "body": _convert_text_to_logs_format(message.body), - "file": _convert_file_to_logs_format(message.file), - }, - ).dict(), - ) - handler = self._lookup_handler_for_exception(exc) - - if handler is None: - exception_logger.exception( - "uncaught {0} exception in handler: {1}", - type(exc).__name__, - exc, - ) - return - - try: - await concurrency.callable_to_coroutine(handler, exc, message) - except Exception as error_handler_exc: - exception_logger.exception( - "uncaught {0} exception in error handler: {1}", - type(error_handler_exc).__name__, - error_handler_exc, - ) - - -def _convert_text_to_logs_format(text: str) -> str: - """Convert text into format that is suitable for logs. - - Arguments: - text: text that should be formatted. - - Returns: - Shape for logging in loguru. - """ - max_log_text_length = 50 - start_text_index = 15 - end_text_index = 5 - - return ( - "...".join((text[:start_text_index], text[-end_text_index:])) - if len(text) > max_log_text_length - else text - ) - - -def _convert_file_to_logs_format(file: Optional[files.File]) -> Optional[dict]: - """Convert file to a new file that will be showed in logs. - - Arguments: - file: file that should be converted. - - Returns: - New file or nothing. - """ - return file.copy(update={"data": "[file content]"}).dict() if file else None diff --git a/botx/middlewares/ns.py b/botx/middlewares/ns.py deleted file mode 100644 index c5005826..00000000 --- a/botx/middlewares/ns.py +++ /dev/null @@ -1,261 +0,0 @@ -"""Definition for middleware that precess next step handlers logic.""" - -import contextlib -from typing import Any, Callable, Dict, List, Optional, Sequence, Tuple, Union -from uuid import UUID - -from loguru import logger -from pydantic import BaseConfig, BaseModel - -from botx import Bot, Collector, concurrency, converters, exceptions -from botx.collecting.handlers.handler import Handler -from botx.collecting.handlers.name_generators import get_name_from_callable -from botx.dependencies.models import Depends -from botx.middlewares.base import BaseMiddleware -from botx.models.messages.message import Message -from botx.typing import Executor - - -class NextStepHandlerState(BaseModel): - """Information about next step handler.""" - - class Config(BaseConfig): - arbitrary_types_allowed = True - - #: name of handler that should be called. - name: str - - #: arguments that should be set on message for handler. - arguments: Dict[str, Any] - - -class NextStepMiddleware(BaseMiddleware): - """ - Naive next step handlers middleware. May be useful in simple apps or as base. - - Important: - This middleware should be the last included into bot, since it will break - execution if right handler will be found. - """ - - def __init__( # noqa: WPS211 - self, - executor: Executor, - bot: Bot, - functions: Union[Dict[str, Callable], Sequence[Callable]], - break_handler: Optional[Union[Handler, str, Callable]] = None, - dependencies: Optional[Sequence[Depends]] = None, - dependency_overrides_provider: Any = None, - ) -> None: - """Init middleware with required query_params. - - Arguments: - executor: next callable that should be executed. - bot: bot that will store ns state. - functions: dict of functions and their names that will be used as next step - handlers or set of sequence of functions that will be registered by - their names. - break_handler: handler instance or name of handler that will break next step - handlers chain. - dependencies: background dependencies that should be applied to handlers. - dependency_overrides_provider: object that will override dependencies for - this handler. - """ - super().__init__(executor) - - dependencies = converters.optional_sequence_to_list( - bot.collector.dependencies, - ) + converters.optional_sequence_to_list( - dependencies, - ) # noqa: W503 - dep_override = ( - dependency_overrides_provider or bot.collector.dependency_overrides_provider - ) - bot.state.ns_collector = Collector( - dependencies=dependencies, - dependency_overrides_provider=dep_override, - ) - bot.state.ns_store = {} - - bot.state.ns_break_handler = None - if break_handler: - if isinstance(break_handler, Handler): - bot.state.ns_collector.handlers.append(break_handler) - bot.state.ns_break_handler = break_handler.name - elif callable(break_handler): - register_function_as_ns_handler(bot, break_handler) - bot.state.ns_break_handler = get_name_from_callable(break_handler) - else: - bot.state.ns_collector.handlers.append( - bot.collector.handler_for(break_handler), - ) - bot.state.ns_break_handler = break_handler - - if isinstance(functions, dict): - functions_dict = functions - else: - functions_dict = {get_name_from_callable(func): func for func in functions} - - for name, function in functions_dict.items(): - register_function_as_ns_handler(bot, function, name) - - async def dispatch(self, message: Message, call_next: Executor) -> None: - """Execute middleware logic. - - Arguments: - message: incoming message. - call_next: next executor in middleware chain. - """ - if message.bot.state.ns_break_handler: - break_handler = message.bot.state.ns_collector.handler_for( - message.bot.state.ns_break_handler, - ) - if break_handler.matches(message): - await self.drop_next_step_handlers_chain(message) - await break_handler(message) - return - - try: - next_handler, state = await self.lookup_next_handler_for_message(message) - except (exceptions.NoMatchFound, IndexError, KeyError, RuntimeError): - await concurrency.callable_to_coroutine(call_next, message) - return - - key = get_chain_key_by_message(message) - logger.bind(botx_ns_middleware=True, payload={"next_step_key": key}).info( - "botx: found next step handler", - ) - - for state_argument in state.arguments.items(): - setattr(message.state, state_argument[0], state_argument[1]) - await next_handler(message) - - async def lookup_next_handler_for_message( - self, - message: Message, - ) -> Tuple[Handler, NextStepHandlerState]: - """Find handler in bot storage or in handlers. - - Arguments: - message: message for which next step handler should be found. - - Returns: - Found handler and state with arguments for message. - """ - handlers: List[NextStepHandlerState] = message.bot.state.ns_store[ - get_chain_key_by_message(message) - ] - handler_state = handlers.pop() - return ( - message.bot.state.ns_collector.handler_for(handler_state.name), - handler_state, - ) - - async def drop_next_step_handlers_chain(self, message: Message) -> None: - """Drop registered chain for message. - - Arguments: - message: message for which chain should be dropped. - """ - with contextlib.suppress(KeyError): - message.bot.state.ns_store.pop(get_chain_key_by_message(message)) - - -def get_chain_key_by_message(message: Message) -> Tuple[str, UUID, UUID, UUID]: - """Generate key for next step handlers chain from message. - - Arguments: - message: message from which key should be generated. - - Returns: - Key using which handler should be found. - - Raises: - RuntimeError: raised if key for chain can not be built. - """ - # key is a tuple of (host, bot_id, chat_id, user_huid) - if not (message.user_huid and message.group_chat_id): - raise RuntimeError("key for chain can be obtained only for messages from users") - - return message.host, message.bot_id, message.group_chat_id, message.user_huid - - -def register_function_as_ns_handler( - bot: Bot, - func: Callable, - name: Optional[str] = None, -) -> None: - """Register new function that can be called as next step handler. - - !!! warning - This functions should not be called to dynamically register new functions in - handlers or elsewhere, since state on different time can be changed somehow. - - Arguments: - bot: bot that stores ns state. - func: functions that will be called as ns handler. Will be transformed to - coroutine if it is not already. - name: name for new function. Will be generated from `func` if not passed. - - Raises: - ValueError: raised if there is error to register handler. - """ - name = name or get_name_from_callable(func) - collector: Collector = bot.state.ns_collector - try: - collector.add_handler( - body=name, - name=name, - handler=func, - include_in_status=False, - dependencies=collector.dependencies, - dependency_overrides_provider=collector.dependency_overrides_provider, - ) - except AssertionError as exc: - raise ValueError(exc.args) - - -def register_next_step_handler( - message: Message, - func: Union[str, Callable], - **ns_arguments: Any, -) -> None: - """Register new next step handler for next message from user. - - !!! info - While registration handler for next message this function fill first try to find - handlers that were registered using `register_function_as_ns_handler`, then - handlers that are registered in bot itself and then if no one was found an - exception will be raised. - - Arguments: - message: incoming message. - func: function name of function which name will be retrieved to register next - handler. - ns_arguments: arguments that will be stored in message state while executing - handler with next message. - - Raises: - ValueError: raised if passed message does not include user_huid or if handler - that should be registered as next step does not exists. - """ - if message.user_huid is None: - raise ValueError( - "message for which ns handler is registered should include user_huid", - ) - - bot = message.bot - collector: Collector = bot.state.ns_collector - name = get_name_from_callable(func) if callable(func) else func - - try: - collector.handler_for(name) - except exceptions.NoMatchFound: - raise ValueError( - "bot does not have registered next step handler with name {0}".format(name), - ) - - key = get_chain_key_by_message(message) - - store: List[NextStepHandlerState] = bot.state.ns_store.setdefault(key, []) - store.append(NextStepHandlerState(name=name, arguments=ns_arguments)) diff --git a/botx/missing.py b/botx/missing.py new file mode 100644 index 00000000..193aacd7 --- /dev/null +++ b/botx/missing.py @@ -0,0 +1,26 @@ +from typing import Any, Literal, TypeVar, Union + + +class _UndefinedType: + """For fields that can be skipped.""" + + def __bool__(self) -> Literal[False]: + return False + + def __repr__(self) -> str: + return "Undefined" + + +RequiredType = TypeVar("RequiredType") +Undefined = _UndefinedType() + +Missing = Union[RequiredType, _UndefinedType] +MissingOptional = Union[RequiredType, None, _UndefinedType] + + +def not_undefined(*args: Any) -> Any: + for arg in args: + if arg is not Undefined: + return arg + + raise ValueError("All arguments have `Undefined` type") diff --git a/botx/models/__init__.py b/botx/models/__init__.py index 1e8935c6..e69de29b 100644 --- a/botx/models/__init__.py +++ b/botx/models/__init__.py @@ -1 +0,0 @@ -"""Pydantic models, data classes or other entities.""" diff --git a/botx/models/api_base.py b/botx/models/api_base.py new file mode 100644 index 00000000..6d219c5c --- /dev/null +++ b/botx/models/api_base.py @@ -0,0 +1,82 @@ +import json +from enum import Enum +from typing import Any, Dict, List, Optional, Set, Union, cast + +from pydantic import BaseModel +from pydantic.json import pydantic_encoder + +from botx.missing import Undefined + + +def _remove_undefined( + origin_obj: Union[Dict[str, Any], List[Any]], +) -> Union[Dict[str, Any], List[Any]]: + if isinstance(origin_obj, dict): + new_dict = {} + + for key, value in origin_obj.items(): + if value is Undefined: + continue + + if isinstance(value, (list, dict)): + new_value = _remove_undefined(value) + if new_value or len(new_value) == len(value): + new_dict[key] = new_value + else: + new_dict[key] = value + + return new_dict + + elif isinstance(origin_obj, list): + new_list = [] + + for value in origin_obj: + if value is Undefined: + continue + + if isinstance(value, (list, dict)): + new_value = _remove_undefined(value) + if new_value or len(new_value) == len(value): + new_list.append(new_value) + else: + new_list.append(value) + + return new_list + + raise NotImplementedError + + +class PayloadBaseModel(BaseModel): + def json(self) -> str: # type: ignore [override] + clean_dict = _remove_undefined(self.dict()) + return json.dumps(clean_dict, default=pydantic_encoder, ensure_ascii=False) + + def jsonable_dict(self) -> Dict[str, Any]: + return cast( + Dict[str, Any], + json.loads(self.json()), + ) + + +class VerifiedPayloadBaseModel(PayloadBaseModel): + """Pydantic base model for API models.""" + + class Config: + use_enum_values = True + + +class UnverifiedPayloadBaseModel(PayloadBaseModel): + def __init__( + self, + _fields_set: Optional[Set[str]] = None, + **kwargs: Any, + ) -> None: + model = BaseModel.construct(_fields_set, **kwargs) + self.__dict__.update(model.__dict__) # noqa: WPS609 (Replace self attrs) + + class Config: + arbitrary_types_allowed = True + + +class StrEnum(str, Enum): # noqa: WPS600 (pydantic needs this inheritance) + """Enum base for API models.""" diff --git a/botx/models/async_files.py b/botx/models/async_files.py new file mode 100644 index 00000000..f7f2fcc1 --- /dev/null +++ b/botx/models/async_files.py @@ -0,0 +1,249 @@ +from contextlib import asynccontextmanager +from dataclasses import dataclass +from typing import AsyncGenerator, Literal, Union, cast +from uuid import UUID + +from aiofiles.tempfile import SpooledTemporaryFile + +from botx.bot.contextvars import bot_id_var, bot_var, chat_id_var +from botx.constants import CHUNK_SIZE +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.enums import ( + APIAttachmentTypes, + AttachmentTypes, + convert_attachment_type_from_domain, + convert_attachment_type_to_domain, +) + + +@dataclass +class AsyncFileBase: + type: AttachmentTypes + filename: str + size: int + + is_async_file: Literal[True] + + _file_id: UUID + _file_url: str + _file_mimetype: str + _file_hash: str + + @asynccontextmanager + async def open(self) -> AsyncGenerator[SpooledTemporaryFile, None]: + bot = bot_var.get() + + async with SpooledTemporaryFile(max_size=CHUNK_SIZE) as tmp_file: + await bot.download_file( + bot_id=bot_id_var.get(), + chat_id=chat_id_var.get(), + file_id=self._file_id, + async_buffer=tmp_file, + ) + + yield tmp_file + + +@dataclass +class Image(AsyncFileBase): + type: Literal[AttachmentTypes.IMAGE] + + +@dataclass +class Video(AsyncFileBase): + type: Literal[AttachmentTypes.VIDEO] + + duration: int + + +@dataclass +class Document(AsyncFileBase): + type: Literal[AttachmentTypes.DOCUMENT] + + +@dataclass +class Voice(AsyncFileBase): + type: Literal[AttachmentTypes.VOICE] + + duration: int + + +class APIAsyncFileBase(VerifiedPayloadBaseModel): + type: APIAttachmentTypes + file: str + file_mime_type: str + file_id: UUID + file_name: str + file_size: int + file_hash: str + + class Config: + """BotX sends extra fields which are used by client only. + + We skip their validation, but extra fields will be saved during + serialization/deserialization. + """ + + extra = "allow" + + +class ApiAsyncFileImage(APIAsyncFileBase): + type: Literal[APIAttachmentTypes.IMAGE] + + +class ApiAsyncFileVideo(APIAsyncFileBase): + type: Literal[APIAttachmentTypes.VIDEO] + + duration: int + + +class ApiAsyncFileDocument(APIAsyncFileBase): + type: Literal[APIAttachmentTypes.DOCUMENT] + + +class ApiAsyncFileVoice(APIAsyncFileBase): + type: Literal[APIAttachmentTypes.VOICE] + + duration: int + + +APIAsyncFile = Union[ + ApiAsyncFileImage, + ApiAsyncFileVideo, + ApiAsyncFileDocument, + ApiAsyncFileVoice, +] + +File = Union[Image, Video, Document, Voice] + + +def convert_async_file_from_domain(file: File) -> APIAsyncFile: + attachment_type = convert_attachment_type_from_domain(file.type) + + if attachment_type == APIAttachmentTypes.IMAGE: + attachment_type = cast(Literal[APIAttachmentTypes.IMAGE], attachment_type) + file = cast(Image, file) + + return ApiAsyncFileImage( + type=attachment_type, + file_name=file.filename, + file_size=file.size, + file_id=file._file_id, + file=file._file_url, + file_mime_type=file._file_mimetype, + file_hash=file._file_hash, + ) + + if attachment_type == APIAttachmentTypes.VIDEO: + attachment_type = cast(Literal[APIAttachmentTypes.VIDEO], attachment_type) + file = cast(Video, file) + + return ApiAsyncFileVideo( + type=attachment_type, + file_name=file.filename, + file_size=file.size, + duration=file.duration, + file_id=file._file_id, + file=file._file_url, + file_mime_type=file._file_mimetype, + file_hash=file._file_hash, + ) + + if attachment_type == APIAttachmentTypes.DOCUMENT: + attachment_type = cast(Literal[APIAttachmentTypes.DOCUMENT], attachment_type) + file = cast(Document, file) + + return ApiAsyncFileDocument( + type=attachment_type, + file_name=file.filename, + file_size=file.size, + file_id=file._file_id, + file=file._file_url, + file_mime_type=file._file_mimetype, + file_hash=file._file_hash, + ) + + if attachment_type == APIAttachmentTypes.VOICE: + attachment_type = cast(Literal[APIAttachmentTypes.VOICE], attachment_type) + file = cast(Voice, file) + + return ApiAsyncFileVoice( + type=attachment_type, + file_name=file.filename, + file_size=file.size, + duration=file.duration, + file_id=file._file_id, + file=file._file_url, + file_mime_type=file._file_mimetype, + file_hash=file._file_hash, + ) + + raise NotImplementedError(f"Unsupported attachment type: {attachment_type}") + + +def convert_async_file_to_domain(async_file: APIAsyncFile) -> File: + attachment_type = convert_attachment_type_to_domain(async_file.type) + + if attachment_type == AttachmentTypes.IMAGE: + attachment_type = cast(Literal[AttachmentTypes.IMAGE], attachment_type) + async_file = cast(ApiAsyncFileImage, async_file) + + return Image( + type=attachment_type, + filename=async_file.file_name, + size=async_file.file_size, + is_async_file=True, + _file_id=async_file.file_id, + _file_mimetype=async_file.file_mime_type, + _file_url=async_file.file, + _file_hash=async_file.file_hash, + ) + + if attachment_type == AttachmentTypes.VIDEO: + attachment_type = cast(Literal[AttachmentTypes.VIDEO], attachment_type) + async_file = cast(ApiAsyncFileVideo, async_file) + + return Video( + type=attachment_type, + filename=async_file.file_name, + size=async_file.file_size, + duration=async_file.duration, + is_async_file=True, + _file_id=async_file.file_id, + _file_mimetype=async_file.file_mime_type, + _file_url=async_file.file, + _file_hash=async_file.file_hash, + ) + + if attachment_type == AttachmentTypes.DOCUMENT: + attachment_type = cast(Literal[AttachmentTypes.DOCUMENT], attachment_type) + async_file = cast(ApiAsyncFileDocument, async_file) + + return Document( + type=attachment_type, + filename=async_file.file_name, + size=async_file.file_size, + is_async_file=True, + _file_id=async_file.file_id, + _file_mimetype=async_file.file_mime_type, + _file_url=async_file.file, + _file_hash=async_file.file_hash, + ) + + if attachment_type == AttachmentTypes.VOICE: + attachment_type = cast(Literal[AttachmentTypes.VOICE], attachment_type) + async_file = cast(ApiAsyncFileVoice, async_file) + + return Voice( + type=attachment_type, + filename=async_file.file_name, + size=async_file.file_size, + duration=async_file.duration, + is_async_file=True, + _file_id=async_file.file_id, + _file_mimetype=async_file.file_mime_type, + _file_url=async_file.file, + _file_hash=async_file.file_hash, + ) + + raise NotImplementedError(f"Unsupported attachment type: {attachment_type}") diff --git a/botx/models/attachments.py b/botx/models/attachments.py index 1a942f6a..3a3d3ab8 100644 --- a/botx/models/attachments.py +++ b/botx/models/attachments.py @@ -1,370 +1,438 @@ -"""Module with attachments for botx.""" +import base64 +from contextlib import asynccontextmanager +from dataclasses import dataclass +from types import MappingProxyType +from typing import AsyncGenerator, Literal, Union, cast -from typing import List, Optional, Union, cast +from aiofiles.tempfile import SpooledTemporaryFile -from pydantic import Field +from botx.async_buffer import AsyncBufferReadable +from botx.constants import CHUNK_SIZE +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.enums import ( + APIAttachmentTypes, + AttachmentTypes, + convert_attachment_type_to_domain, +) -from botx.models.base import BotXBaseModel -from botx.models.enums import AttachmentsTypes, LinkProtos -from botx.models.files import File -try: - from typing import Literal # noqa: WPS433 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 +@dataclass +class FileAttachmentBase: + type: AttachmentTypes + filename: str + size: int + is_async_file: Literal[False] -class FileAttachment(BotXBaseModel): - """Class that represents file in RFC 2397 format.""" + content: bytes - #: name of file. - file_name: Optional[str] - - #: file content in RFC 2397 format. - content: str + @asynccontextmanager + async def open(self) -> AsyncGenerator[SpooledTemporaryFile, None]: + async with SpooledTemporaryFile(max_size=CHUNK_SIZE) as tmp_file: + await tmp_file.write(self.content) + await tmp_file.seek(0) + yield tmp_file -class Image(FileAttachment): - """Image model from botx.""" - file_name: str = "image.jpg" +@dataclass +class AttachmentImage(FileAttachmentBase): + type: Literal[AttachmentTypes.IMAGE] -class Video(FileAttachment): - """Video model from botx.""" +@dataclass +class AttachmentVideo(FileAttachmentBase): + type: Literal[AttachmentTypes.VIDEO] - file_name: str = "video.mp4" - - #: video duration duration: int -class Document(FileAttachment): - """Document model from botx.""" - - file_name: str = "document.docx" +@dataclass +class AttachmentDocument(FileAttachmentBase): + type: Literal[AttachmentTypes.DOCUMENT] -class Voice(BotXBaseModel): - """Voice model from botx.""" +@dataclass +class AttachmentVoice(FileAttachmentBase): + type: Literal[AttachmentTypes.VOICE] - #: file content in RFC 2397 format. - content: str - - #: voice duration duration: int -class Location(BotXBaseModel): - """Location model from botx.""" - - #: name of location - location_name: str - - #: address of location - location_address: str +@dataclass +class AttachmentLocation: + type: Literal[AttachmentTypes.LOCATION] - #: latitude of location - location_lat: float + name: str + address: str + latitude: str + longitude: str - #: longitude of location - location_lng: float +@dataclass +class AttachmentContact: + type: Literal[AttachmentTypes.CONTACT] -class Contact(BotXBaseModel): - """Contact model from botx.""" + name: str - #: name of contact - contact_name: str +@dataclass +class AttachmentLink: + type: Literal[AttachmentTypes.LINK] -class Link(BotXBaseModel): - """Class that marked as Link from botx.""" - - #: url of link url: str + title: str + preview: str + text: str - #: title of url - url_title: Optional[str] = None - - #: link on preview of this link - url_preview: Optional[str] = None - - #: text on preview - url_text: Optional[str] = None - def is_mail(self) -> bool: - """Confirm that is email link.""" - return self.url.startswith(LinkProtos.email) - - def is_telephone(self) -> bool: - """Confirm that is telephone link.""" - return self.url.startswith(LinkProtos.telephone) - - def is_link(self) -> bool: - """Confirm that is link on resource.""" - return not (self.is_mail() or self.is_telephone()) +IncomingFileAttachment = Union[ + AttachmentImage, + AttachmentVideo, + AttachmentDocument, + AttachmentVoice, +] - @property - def mailto(self) -> str: - """Property that retuning email address without protocol.""" - if not self.is_mail(): - raise AttributeError("mailto") - return self.url[len(LinkProtos.email) :] # noqa: E203 - @property - def tel(self) -> str: - """Property that retuning telephone number without protocol.""" - if not self.is_telephone(): - raise AttributeError("telephone number") - return self.url[len(LinkProtos.telephone) :] # noqa: E203 +@dataclass +class OutgoingAttachment: + content: bytes + filename: str + is_async_file: Literal[False] = False + @classmethod + async def from_async_buffer( + cls, + async_buffer: AsyncBufferReadable, + filename: str, + ) -> "OutgoingAttachment": + return cls( + content=await async_buffer.read(), + filename=filename, + ) -Attachments = Union[Image, Video, Document, Voice, Location, Contact, Link] +class BotAPIAttachmentImageData(VerifiedPayloadBaseModel): + content: str + file_name: str -class ImageAttachment(BotXBaseModel): - """BotX API image attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.image] = Field(default=AttachmentsTypes.image) +class BotAPIAttachmentImage(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.IMAGE] + data: BotAPIAttachmentImageData - #: content of attachment - data: Image +class BotAPIAttachmentVideoData(VerifiedPayloadBaseModel): + content: str + file_name: str + duration: int -class VideoAttachment(BotXBaseModel): - """BotX API video attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.video] = Field(default=AttachmentsTypes.video) +class BotAPIAttachmentVideo(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.VIDEO] + data: BotAPIAttachmentVideoData - #: content of attachment - data: Video +class BotAPIAttachmentDocumentData(VerifiedPayloadBaseModel): + content: str + file_name: str -class DocumentAttachment(BotXBaseModel): - """BotX API document attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.document] = Field(default=AttachmentsTypes.document) +class BotAPIAttachmentDocument(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.DOCUMENT] + data: BotAPIAttachmentDocumentData - #: content of attachment - data: Document +class BotAPIAttachmentVoiceData(VerifiedPayloadBaseModel): + content: str + duration: int -class VoiceAttachment(BotXBaseModel): - """BotX API voice attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.voice] = Field(default=AttachmentsTypes.voice) +class BotAPIAttachmentVoice(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.VOICE] + data: BotAPIAttachmentVoiceData - #: content of attachment - data: Voice +class BotAPIAttachmentLocationData(VerifiedPayloadBaseModel): + location_name: str + location_address: str + location_lat: str + location_lng: str -class ContactAttachment(BotXBaseModel): - """BotX API contact attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.contact] = Field(default=AttachmentsTypes.contact) +class BotAPIAttachmentLocation(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.LOCATION] + data: BotAPIAttachmentLocationData - #: content of attachment - data: Contact +class BotAPIAttachmentContactData(VerifiedPayloadBaseModel): + contact_name: str -class LocationAttachment(BotXBaseModel): - """BotX API location attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.location] = Field(default=AttachmentsTypes.location) +class BotAPIAttachmentContact(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.CONTACT] + data: BotAPIAttachmentContactData - #: content of attachment - data: Location +class BotAPIAttachmentLinkData(VerifiedPayloadBaseModel): + url: str + url_title: str + url_preview: str + url_text: str -class LinkAttachment(BotXBaseModel): - """BotX API link attachment container.""" - #: type of attachment - type: Literal[AttachmentsTypes.link] = Field(default=AttachmentsTypes.link) +class BotAPIAttachmentLink(VerifiedPayloadBaseModel): + type: Literal[APIAttachmentTypes.LINK] + data: BotAPIAttachmentLinkData - #: content of attachment - data: Link +BotAPIAttachment = Union[ + BotAPIAttachmentVideo, + BotAPIAttachmentImage, + BotAPIAttachmentDocument, + BotAPIAttachmentVoice, + BotAPIAttachmentLocation, + BotAPIAttachmentContact, + BotAPIAttachmentLink, +] -Attachment = Union[ - ImageAttachment, - VideoAttachment, - DocumentAttachment, - VoiceAttachment, - ContactAttachment, - LocationAttachment, - LinkAttachment, +IncomingAttachment = Union[ + IncomingFileAttachment, + AttachmentLocation, + AttachmentContact, + AttachmentLink, ] -class AttachList(BotXBaseModel): # noqa: WPS214, WPS338 - """Additional wrapped class for use property.""" - - __root__: List[Attachment] - - def _get_attach_by_type(self, attach_type: AttachmentsTypes) -> Attachments: - for attach in self.all_attachments: - if attach.type == attach_type: - return attach.data - raise AttributeError(attach_type) - - @property - def image(self) -> Image: - """Parse attachments. - - Returns: - Image: image from attachments. - Raises: - AttributeError: message has no image. - """ - attach = self._get_attach_by_type(AttachmentsTypes.image) - return cast(Image, attach) - - @property - def document(self) -> Document: - """Parse attachments. - - Returns: - Document: document from attachments. - Raises: - AttributeError: message has no document. - """ - attach = self._get_attach_by_type(AttachmentsTypes.document) - return cast(Document, attach) - - @property - def location(self) -> Location: - """Parse attachments. - - Returns: - Location: location from attachments. - Raises: - AttributeError: message has no location. - """ - attach = self._get_attach_by_type(AttachmentsTypes.location) - return cast(Location, attach) - - @property - def contact(self) -> Contact: - """Parse attachments. - - Returns: - Contact: contact from attachments. - Raises: - AttributeError: message has no contact. - """ - attach = self._get_attach_by_type(AttachmentsTypes.contact) - return cast(Contact, attach) - - @property - def voice(self) -> Voice: - """Parse attachments. - - Returns: - Voice: voice from attachments - Raises: - AttributeError: message has no voice. - """ - attach = self._get_attach_by_type(AttachmentsTypes.voice) - return cast(Voice, attach) - - @property - def video(self) -> Video: - """Parse attachments. - - Returns: - Video: video from attachments. - Raises: - AttributeError: message has no video. - """ - attach = self._get_attach_by_type(AttachmentsTypes.video) - return cast(Video, attach) - - @property - def link(self) -> Link: - """Parse attachments. - - Returns: - Link: lint to resource from attachments. - Raises: - AttributeError: message has no link. - """ - attach = self._get_attach_by_type(AttachmentsTypes.link) - if attach.is_link(): # type: ignore - return cast(Link, attach) - raise AttributeError(AttachmentsTypes.link) - - @property - def email(self) -> str: - """Parse attachments. - - Returns: - str: email from attachments. - Raises: - AttributeError: message has no email. - """ - attach = self._get_attach_by_type(AttachmentsTypes.link) - if attach.is_mail(): # type: ignore - return attach.mailto # type: ignore - raise AttributeError(AttachmentsTypes.link) - - @property - def telephone(self) -> str: - """Parse attachments. - - Returns: - str: telephone number from attachments. - Raises: - AttributeError: message has no telephone. - """ - attach = self._get_attach_by_type(AttachmentsTypes.link) - if attach.is_telephone(): # type: ignore - return attach.tel # type: ignore - raise AttributeError(AttachmentsTypes.link) - - @property - def all_attachments(self) -> List[Attachment]: - """Search attachments in message. - - Returns: - List of attachments. - """ - return self.__root__ - - @property - def file(self) -> File: - """Search file in message's attachments. - - Returns: - Botx file from video, image or document. - Raises: - AttributeError: message has no file. - """ - for attachment in self.all_attachments: - if isinstance(attachment.data, FileAttachment): - return File.construct( - file_name=attachment.data.file_name, - data=attachment.data.content, - ) - raise AttributeError - - @property - def attach_type(self) -> AttachmentsTypes: - """Get attachment type. - - Returns: - AttachmentsTypes: Attachment type. - Raises: - AttributeError: message has no attachment. - """ - if self.all_attachments: - return self.all_attachments[0].type - - raise AttributeError +def convert_api_attachment_to_domain( # noqa: WPS212 + api_attachment: BotAPIAttachment, +) -> IncomingAttachment: + attachment_type = convert_attachment_type_to_domain(api_attachment.type) + + if attachment_type == AttachmentTypes.IMAGE: + attachment_type = cast(Literal[AttachmentTypes.IMAGE], attachment_type) + api_attachment = cast(BotAPIAttachmentImage, api_attachment) + content = decode_rfc2397(api_attachment.data.content) + + return AttachmentImage( + type=attachment_type, + filename=api_attachment.data.file_name, + size=len(content), + is_async_file=False, + content=content, + ) + + if attachment_type == AttachmentTypes.VIDEO: + attachment_type = cast(Literal[AttachmentTypes.VIDEO], attachment_type) + api_attachment = cast(BotAPIAttachmentVideo, api_attachment) + content = decode_rfc2397(api_attachment.data.content) + + return AttachmentVideo( + type=attachment_type, + filename=api_attachment.data.file_name, + size=len(content), + is_async_file=False, + content=content, + duration=api_attachment.data.duration, + ) + + if attachment_type == AttachmentTypes.DOCUMENT: + attachment_type = cast(Literal[AttachmentTypes.DOCUMENT], attachment_type) + api_attachment = cast(BotAPIAttachmentDocument, api_attachment) + content = decode_rfc2397(api_attachment.data.content) + + return AttachmentDocument( + type=attachment_type, + filename=api_attachment.data.file_name, + size=len(content), + is_async_file=False, + content=content, + ) + + if attachment_type == AttachmentTypes.VOICE: + attachment_type = cast(Literal[AttachmentTypes.VOICE], attachment_type) + api_attachment = cast(BotAPIAttachmentVoice, api_attachment) + content = decode_rfc2397(api_attachment.data.content) + + return AttachmentVoice( + type=attachment_type, + filename="record.mp3", + size=len(content), + is_async_file=False, + content=content, + duration=api_attachment.data.duration, + ) + + if attachment_type == AttachmentTypes.LOCATION: + attachment_type = cast(Literal[AttachmentTypes.LOCATION], attachment_type) + api_attachment = cast(BotAPIAttachmentLocation, api_attachment) + + return AttachmentLocation( + type=attachment_type, + name=api_attachment.data.location_name, + address=api_attachment.data.location_address, + latitude=api_attachment.data.location_lat, + longitude=api_attachment.data.location_lng, + ) + + if attachment_type == AttachmentTypes.CONTACT: + attachment_type = cast(Literal[AttachmentTypes.CONTACT], attachment_type) + api_attachment = cast(BotAPIAttachmentContact, api_attachment) + + return AttachmentContact( + type=attachment_type, + name=api_attachment.data.contact_name, + ) + + if attachment_type == AttachmentTypes.LINK: + attachment_type = cast(Literal[AttachmentTypes.LINK], attachment_type) + api_attachment = cast(BotAPIAttachmentLink, api_attachment) + + return AttachmentLink( + type=attachment_type, + url=api_attachment.data.url, + title=api_attachment.data.url_title, + preview=api_attachment.data.url_preview, + text=api_attachment.data.url_text, + ) + + raise NotImplementedError(f"Unsupported attachment type: {attachment_type}") + + +def decode_rfc2397(encoded_content: str) -> bytes: + # "" -> b"hello" + return base64.b64decode(encoded_content.split(",", 1)[1].encode()) + + +EXTENSIONS_TO_MIMETYPES = MappingProxyType( + { + # application + "7z": "application/x-7z-compressed", + "abw": "application/x-abiword", + "ai": "application/postscript", + "arc": "application/x-freearc", + "azw": "application/vnd.amazon.ebook", + "bin": "application/octet-stream", + "bz": "application/x-bzip", + "bz2": "application/x-bzip2", + "cda": "application/x-cdf", + "csh": "application/x-csh", + "doc": "application/msword", + "docx": ( + "application/vnd.openxmlformats-officedocument.wordprocessingml.document" + ), + "eot": "application/vnd.ms-fontobject", + "eps": "application/postscript", + "epub": "application/epub+zip", + "gz": "application/gzip", + "jar": "application/java-archive", + "json-api": "application/vnd.api+json", + "json-patch": "application/json-patch+json", + "json": "application/json", + "jsonld": "application/ld+json", + "mdb": "application/x-msaccess", + "mpkg": "application/vnd.apple.installer+xml", + "odp": "application/vnd.oasis.opendocument.presentation", + "ods": "application/vnd.oasis.opendocument.spreadsheet", + "odt": "application/vnd.oasis.opendocument.text", + "ogx": "application/ogg", + "pdf": "application/pdf", + "php": "application/x-httpd-php", + "ppt": "application/vnd.ms-powerpoint", + "pptx": ( + "application/vnd.openxmlformats-officedocument.presentationml.presentation" + ), + "ps": "application/postscript", + "rar": "application/vnd.rar", + "rtf": "application/rtf", + "sh": "application/x-sh", + "swf": "application/x-shockwave-flash", + "tar": "application/x-tar", + "vsd": "application/vnd.visio", + "wasm": "application/wasm", + "webmanifest": "application/manifest+json", + "xhtml": "application/xhtml+xml", + "xls": "application/vnd.ms-excel", + "xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", + "xul": "application/vnd.mozilla.xul+xml", + "zip": "application/zip", + # audio + "aac": "audio/aac", + "mid": "audio/midi", + "midi": "audio/midi", + "mp3": "audio/mpeg", + "oga": "audio/ogg", + "opus": "audio/opus", + "wav": "audio/wav", + "weba": "audio/webm", + # font + "otf": "font/otf", + "ttf": "font/ttf", + "woff": "font/woff", + "woff2": "font/woff2", + # image + "avif": "image/avif", + "bmp": "image/bmp", + "gif": "image/gif", + "ico": "image/vnd.microsoft.icon", + "jpeg": "image/jpeg", + "jpg": "image/jpeg", + "png": "image/png", + "svg": "image/svg+xml", + "svgz": "image/svg+xml", + "tif": "image/tiff", + "tiff": "image/tiff", + "webp": "image/webp", + # text + "css": "text/css", + "csv": "text/csv", + "htm": "text/html", + "html": "text/html", + "ics": "text/calendar", + "js": "text/javascript", + "mjs": "text/javascript", + "txt": "text/plain", + "text": "text/plain", + "xml": "text/xml", + # video + "3g2": "video/3gpp2", + "3gp": "video/3gpp", + "avi": "video/x-msvideo", + "mov": "video/quicktime", + "mp4": "video/mp4", + "mpeg": "video/mpeg", + "mpg": "video/mpeg", + "ogv": "video/ogg", + "ts": "video/mp2t", + "webm": "video/webm", + "wmv": "video/x-ms-wmv", + }, +) +DEFAULT_MIMETYPE = "application/octet-stream" + + +def encode_rfc2397(content: bytes, mimetype: str) -> str: + b64_content = base64.b64encode(content).decode() + return f"data:{mimetype};base64,{b64_content}" + + +class BotXAPIAttachment(UnverifiedPayloadBaseModel): + file_name: str + data: str + + @classmethod + def from_file_attachment( + cls, + attachment: Union[IncomingFileAttachment, OutgoingAttachment], + ) -> "BotXAPIAttachment": + assert attachment.content is not None + + mimetype = EXTENSIONS_TO_MIMETYPES.get( + attachment.filename.split(".")[-1], + DEFAULT_MIMETYPE, + ) + + return cls( + file_name=attachment.filename, + data=encode_rfc2397(attachment.content, mimetype), + ) diff --git a/botx/models/attachments_meta.py b/botx/models/attachments_meta.py deleted file mode 100644 index 7c2469b9..00000000 --- a/botx/models/attachments_meta.py +++ /dev/null @@ -1,87 +0,0 @@ -"""Module with attachments meta for botx.""" - -from typing import Optional, Union - -from pydantic import Field - -from botx.models.base import BotXBaseModel -from botx.models.enums import AttachmentsTypes - -try: - from typing import Literal # noqa: WPS433 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 - - -class FileAttachmentMeta(BotXBaseModel): - """Common metadata of file.""" - - #: type of attachment - type: str - - #: name of file. - file_name: str - - #: mime type of file - file_mime_type: Optional[str] - - #: file preview in RFC 2397 format. - file_preview_base64: Optional[str] - - -class ImageAttachmentMeta(FileAttachmentMeta): - """BotX API image attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.image] = Field(default=AttachmentsTypes.image) - - -class VideoAttachmentMeta(FileAttachmentMeta): - """BotX API video attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.video] = Field(default=AttachmentsTypes.video) - - -class DocumentAttachmentMeta(FileAttachmentMeta): - """BotX API document attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.document] = Field(default=AttachmentsTypes.document) - - -class VoiceAttachmentMeta(FileAttachmentMeta): - """BotX API voice attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.voice] = Field(default=AttachmentsTypes.voice) - - -class ContactAttachmentMeta(BotXBaseModel): - """BotX API contact attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.contact] = Field(default=AttachmentsTypes.contact) - - #: name of contact - contact_name: str - - -class LocationAttachmentMeta(BotXBaseModel): - """BotX API location attachment meta container.""" - - #: type of attachment - type: Literal[AttachmentsTypes.location] = Field(default=AttachmentsTypes.location) - - #: address of location - location_address: str - - -AttachmentMeta = Union[ - ImageAttachmentMeta, - VideoAttachmentMeta, - DocumentAttachmentMeta, - VoiceAttachmentMeta, - ContactAttachmentMeta, - LocationAttachmentMeta, -] diff --git a/botx/models/base.py b/botx/models/base.py deleted file mode 100644 index 95c23d9a..00000000 --- a/botx/models/base.py +++ /dev/null @@ -1,9 +0,0 @@ -"""Module with base model classes.""" -from pydantic import BaseModel - - -class BotXBaseModel(BaseModel): - """Base class for configure all models.""" - - class Config: - use_enum_values = True diff --git a/botx/models/base_command.py b/botx/models/base_command.py new file mode 100644 index 00000000..8e7963ce --- /dev/null +++ b/botx/models/base_command.py @@ -0,0 +1,73 @@ +from dataclasses import dataclass +from typing import Any, Dict, Literal, Optional +from uuid import UUID + +from pydantic import validator + +from botx.bot.api.exceptions import UnsupportedBotAPIVersionError +from botx.constants import BOT_API_VERSION +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.bot_account import BotAccount +from botx.models.enums import APIChatTypes, BotAPIClientPlatforms, BotAPICommandTypes + + +class BotAPICommandPayload(VerifiedPayloadBaseModel): + body: str + command_type: Literal[BotAPICommandTypes.USER] + data: Dict[str, Any] + metadata: Dict[str, Any] + + +class BotAPIDeviceMeta(VerifiedPayloadBaseModel): + pushes: Optional[bool] + timezone: Optional[str] + permissions: Optional[Dict[str, Any]] + + +class BaseBotAPIContext(VerifiedPayloadBaseModel): + host: str + + +class BotAPIUserContext(BaseBotAPIContext): + user_huid: UUID + ad_domain: Optional[str] + ad_login: Optional[str] + username: Optional[str] + is_admin: Optional[bool] + is_creator: Optional[bool] + + +class BotAPIChatContext(BaseBotAPIContext): + group_chat_id: UUID + chat_type: APIChatTypes + + +class BotAPIDeviceContext(BaseBotAPIContext): + app_version: Optional[str] + platform: Optional[BotAPIClientPlatforms] + platform_package_id: Optional[str] + device: Optional[str] + device_meta: Optional[BotAPIDeviceMeta] + device_software: Optional[str] + manufacturer: Optional[str] + locale: Optional[str] + + +class BotAPIBaseCommand(VerifiedPayloadBaseModel): + bot_id: UUID + sync_id: UUID + proto_version: int + + @validator("proto_version", pre=True) + @classmethod + def validate_proto_version(cls, version: Any) -> int: + if isinstance(version, int) and version == BOT_API_VERSION: + return version + + raise UnsupportedBotAPIVersionError(version) + + +@dataclass +class BotCommandBase: + bot: BotAccount + raw_command: Optional[Dict[str, Any]] diff --git a/botx/models/bot_account.py b/botx/models/bot_account.py new file mode 100644 index 00000000..5c201ea7 --- /dev/null +++ b/botx/models/bot_account.py @@ -0,0 +1,13 @@ +from dataclasses import dataclass +from uuid import UUID + + +@dataclass +class BotAccount: + id: UUID + host: str + + +@dataclass +class BotAccountWithSecret(BotAccount): + secret_key: str diff --git a/botx/models/bot_sender.py b/botx/models/bot_sender.py new file mode 100644 index 00000000..5709c7ad --- /dev/null +++ b/botx/models/bot_sender.py @@ -0,0 +1,10 @@ +from dataclasses import dataclass +from typing import Optional +from uuid import UUID + + +@dataclass +class BotSender: + huid: UUID + is_chat_admin: Optional[bool] + is_chat_creator: Optional[bool] diff --git a/botx/models/buttons.py b/botx/models/buttons.py deleted file mode 100644 index 8ba2137a..00000000 --- a/botx/models/buttons.py +++ /dev/null @@ -1,90 +0,0 @@ -"""Pydantic models for bubbles and keyboard buttons.""" - -from typing import Optional - -from pydantic import validator - -from botx.models.base import BotXBaseModel -from botx.models.enums import ButtonHandlerTypes - - -class ButtonOptions(BotXBaseModel): - """Extra options for buttons, like disabling output by tap.""" - - #: if True then text won't shown for user in messenger. - silent: bool = True - - #: button width weight (the more weight, the more occupied space). - h_size: int = 1 - - #: show toast with `alert_text` when user press the button - show_alert: bool = False - - #: text to be shown in toast (show command body if `alert_text` is `None`). - alert_text: Optional[str] = None - - #: platform, that handle command from markup. If `bot` - command should be send - # to bot, else(`client`) should be executed by client. - handler: ButtonHandlerTypes = ButtonHandlerTypes.bot - - @validator("h_size") - def h_size_should_be_positive(cls, h_size: int) -> int: # noqa: N805 - """Validate that `h_size` is positive integer. - - Arguments: - h_size: width weight for validation. - - Returns: - Validated `h_size`. - - Raises: - ValueError: if `h_size` is not valid. - """ - if h_size < 1: - raise ValueError("h_size should be positive integer") - - return h_size - - -class Button(BotXBaseModel): - """Base class for ui element like bubble or keyboard button.""" - - #: command that will be triggered by click on the element. - command: str - - #: text that will be shown on the element. - label: Optional[str] = None - - #: extra payload that will be stored in button and then received in new message. - data: dict = {} # noqa: WPS110 - - #: options for button. - opts: ButtonOptions = ButtonOptions() - - @validator("label", always=True) - def label_as_command_if_none( - cls, - label: Optional[str], - values: dict, # noqa: N805, WPS110 - ) -> str: - """Return command as label if it is `None`. - - Arguments: - label: value that should be checked. - values: all other validated_values checked before. - - Returns: - Label for button. - """ - if label is None: - return values["command"] - - return label - - -class BubbleElement(Button): - """Bubble buttons that is shown under messages.""" - - -class KeyboardElement(Button): - """Keyboard buttons that are placed instead of real keyboard.""" diff --git a/botx/models/chats.py b/botx/models/chats.py index 7586808d..9c12c0eb 100644 --- a/botx/models/chats.py +++ b/botx/models/chats.py @@ -1,75 +1,74 @@ -"""Entities for chats.""" - +from dataclasses import dataclass from datetime import datetime -from typing import Iterator, List, Optional +from datetime import datetime as dt +from typing import List, Optional from uuid import UUID -from botx.models.base import BotXBaseModel -from botx.models.enums import ChatTypes -from botx.models.users import UserFromChatSearch - - -class ChatFromSearch(BotXBaseModel): - """Chat from search request.""" +from botx.models.enums import ChatTypes, UserKinds - #: name of chat. - name: str - #: description of chat - description: Optional[str] +@dataclass +class Chat: + id: UUID + type: ChatTypes - #: type of chat. - chat_type: ChatTypes - #: HUID of chat creator. - creator: UUID +@dataclass +class ChatListItem: + """Chat from list. - #: ID of chat. - group_chat_id: UUID + Attributes: + chat_id: Chat id. + chat_type: Chat Type. + name: Chat name. + description: Chat description. + members: Chat members. + created_at: Chat creation datetime. + updated_at: Last chat update datetime. + """ - #: users in chat. - members: List[UserFromChatSearch] - - #: creation datetime of chat. - inserted_at: datetime - - -class BotChatFromList(BotXBaseModel): - """Chat from list.""" - - #: name of chat. + chat_id: UUID + chat_type: ChatTypes name: str - - #: description of chat. description: Optional[str] - - #: type of chat. - chat_type: ChatTypes - - #: ID of chat. - group_chat_id: UUID - - #: users in chat. members: List[UUID] + created_at: datetime + updated_at: datetime - #: datetime bot joined in chat. - inserted_at: datetime - #: update datetime of chat. - updated_at: datetime +@dataclass +class ChatInfoMember: + """Chat member. + Attributes: + is_admin: Is user admin. + huid: User huid. + kind: User type. + """ -class BotChatList(BotXBaseModel): - """Bot's chat list response model.""" + is_admin: bool + huid: UUID + kind: UserKinds - __root__: List[BotChatFromList] - def __iter__(self) -> Iterator[BotChatFromList]: # type: ignore - """Override iterator for pydantic model.""" - return iter(self.__root__) +@dataclass +class ChatInfo: + """Chat information. - def __len__(self) -> int: # noqa: D105 - return len(self.__root__) + Attributes: + chat_type: Chat type. + creator_id: Chat creator id. + description: Chat description. + chat_id: Chat id. + created_at: Chat creation datetime. + members: Chat members. + name: Chat name. + """ - def __getitem__(self, key: int) -> BotChatFromList: # noqa: D105 - return self.__root__[key] + chat_type: ChatTypes + creator_id: UUID + description: Optional[str] + chat_id: UUID + created_at: dt + members: List[ChatInfoMember] + name: str diff --git a/botx/models/commands.py b/botx/models/commands.py new file mode 100644 index 00000000..acd6e64d --- /dev/null +++ b/botx/models/commands.py @@ -0,0 +1,44 @@ +from typing import Union + +from botx.models.message.incoming_message import BotAPIIncomingMessage, IncomingMessage +from botx.models.system_events.added_to_chat import AddedToChatEvent, BotAPIAddedToChat +from botx.models.system_events.chat_created import BotAPIChatCreated, ChatCreatedEvent +from botx.models.system_events.cts_login import BotAPICTSLogin, CTSLoginEvent +from botx.models.system_events.cts_logout import BotAPICTSLogout, CTSLogoutEvent +from botx.models.system_events.deleted_from_chat import ( + BotAPIDeletedFromChat, + DeletedFromChatEvent, +) +from botx.models.system_events.internal_bot_notification import ( + BotAPIInternalBotNotification, + InternalBotNotificationEvent, +) +from botx.models.system_events.left_from_chat import ( + BotAPILeftFromChat, + LeftFromChatEvent, +) +from botx.models.system_events.smartapp_event import BotAPISmartAppEvent, SmartAppEvent + +BotAPISystemEvent = Union[ + BotAPIChatCreated, + BotAPIAddedToChat, + BotAPIDeletedFromChat, + BotAPILeftFromChat, + BotAPICTSLogin, + BotAPICTSLogout, + BotAPIInternalBotNotification, + BotAPISmartAppEvent, +] +BotAPICommand = Union[BotAPIIncomingMessage, BotAPISystemEvent] + +SystemEvent = Union[ + ChatCreatedEvent, + AddedToChatEvent, + DeletedFromChatEvent, + LeftFromChatEvent, + CTSLoginEvent, + CTSLogoutEvent, + InternalBotNotificationEvent, + SmartAppEvent, +] +BotCommand = Union[IncomingMessage, SystemEvent] diff --git a/botx/models/constants.py b/botx/models/constants.py deleted file mode 100644 index 01451fcd..00000000 --- a/botx/models/constants.py +++ /dev/null @@ -1,3 +0,0 @@ -"""Definition of different constants that are used in models.""" - -MAXIMUM_TEXT_LENGTH = 4096 diff --git a/botx/models/credentials.py b/botx/models/credentials.py deleted file mode 100644 index 35ea7edd..00000000 --- a/botx/models/credentials.py +++ /dev/null @@ -1,40 +0,0 @@ -"""Definition of credentials that are used for access to BotX API.""" - -import base64 -import hashlib -import hmac -from typing import Optional -from uuid import UUID - -from botx.models.base import BotXBaseModel - - -class BotXCredentials(BotXBaseModel): - """Credentials to bot account in express.""" - - #: host name of server. - host: str - - #: secret that will be used for generating signature for bot. - secret_key: str - - #: bot that retrieved token from API. - bot_id: UUID - - #: token generated for bot. - token: Optional[str] = None - - @property - def signature(self) -> str: - """Calculate signature for obtaining token for bot from BotX API. - - Returns: - Calculated signature. - """ - return base64.b16encode( - hmac.new( - key=self.secret_key.encode(), - msg=str(self.bot_id).encode(), - digestmod=hashlib.sha256, - ).digest(), - ).decode() diff --git a/botx/models/datastructures.py b/botx/models/datastructures.py deleted file mode 100644 index de69c87b..00000000 --- a/botx/models/datastructures.py +++ /dev/null @@ -1,44 +0,0 @@ -"""Entities that represent some structs that are used in this library.""" - -from typing import Any, Optional - - -class State: - """An object that can be used to store arbitrary state.""" - - _state: dict - - def __init__(self, state: Optional[dict] = None): - """Init state with required query_params. - - Arguments: - state: initial state. - """ - state = state or {} - super().__setattr__("_state", state) # noqa: WPS613 - - def __setattr__(self, key: Any, new_value: Any) -> None: - """Set state attribute. - - Arguments: - key: key to set attribute. - new_value: value of attribute. - """ - self._state[key] = new_value - - def __getattr__(self, key: Any) -> Any: - """Get state attribute. - - Arguments: - key: key of retrieved attribute. - - Returns: - Stored value. - - Raises: - AttributeError: raised if attribute was not found in state. - """ - try: - return self._state[key] - except KeyError: - raise AttributeError("state has no attribute '{0}'".format(key)) diff --git a/botx/models/entities.py b/botx/models/entities.py deleted file mode 100644 index 232ae68b..00000000 --- a/botx/models/entities.py +++ /dev/null @@ -1,294 +0,0 @@ -"""Entities that can be received in message.""" - -from datetime import datetime -from typing import Dict, List, Optional, Union, cast -from uuid import UUID, uuid4 - -from pydantic import Field, validator - -from botx.models.attachments_meta import AttachmentMeta -from botx.models.base import BotXBaseModel -from botx.models.enums import ChatTypes, EntityTypes, MentionTypes - - -class Forward(BotXBaseModel): - """Forward in message.""" - - #: ID of chat from which forward received. - group_chat_id: UUID - - #: ID of user that is author of message. - sender_huid: UUID - - #: type of forward. - forward_type: ChatTypes - - #: name of original chat. - source_chat_name: Optional[str] = None - - #: id of original message event. - source_sync_id: UUID - - #: id of event creation. - source_inserted_at: datetime - - -class UserMention(BotXBaseModel): - """Mention for single user in chat or by `user_huid`.""" - - #: huid of user that will be mentioned. - user_huid: UUID - - #: name that will be used instead of default user name. - name: Optional[str] = None - - #: connection type via that entity was mention - conn_type: Optional[str] = None - - -class ChatMention(BotXBaseModel): - """Mention chat in message by `group_chat_id`.""" - - #: id of chat that will be mentioned. - group_chat_id: UUID - - #: name that will be used instead of default chat name. - name: Optional[str] = None - - -class Mention(BotXBaseModel): - """Mention that is used in bot in messages.""" - - #: unique id of mention. - mention_id: Optional[UUID] = None - - #: information about mention object - mention_data: Optional[Union[ChatMention, UserMention, Dict]] - - #: payload with data about mention. - mention_type: MentionTypes = MentionTypes.user - - @validator("mention_id", pre=True, always=True) - def generate_mention_id(cls, mention_id: Optional[UUID]) -> UUID: # noqa: N805 - """Verify that `mention_id` will be in mention. - - Arguments: - mention_id: id that should present or new UUID4 will be generated. - - Returns: - Mention ID. - """ - return mention_id or uuid4() - - @validator("mention_data", pre=True, always=True) - def ignore_empty_data( - cls, - mention_data: Union[ChatMention, UserMention, Dict], # noqa: N805 - ) -> Optional[Union[ChatMention, UserMention, Dict]]: - """Pass empty dict into mention_data as None. - - Arguments: - mention_data: dict of mention's data. - - Returns: - Mention's data if is not empty or None. - """ - if mention_data == {}: # noqa: WPS520 - return None - - return mention_data - - @validator("mention_type", pre=True, always=True) - def check_that_type_matches_data( # noqa: WPS231, WPS210 - cls, - mention_type: MentionTypes, - values: dict, # noqa: N805, WPS110 - ) -> MentionTypes: - """Verify that `mention_type` matches provided `mention_data`. - - Arguments: - mention_type: mention type that should be consistent with data. - values: verified data. - - Returns: - Checked mention type. - - Raises: - ValueError: raised if mention_type does not corresponds with data. - """ - mention_data = values.get("mention_data") - if (mention_type != MentionTypes.all_members) and (mention_data is None): - raise ValueError("no `mention_data`, perhaps this entity isn't a mention") - - user_mention_types = {MentionTypes.user, MentionTypes.contact} - chat_mention_types = {MentionTypes.chat, MentionTypes.channel} - - is_user_mention_signature = isinstance(mention_data, UserMention) and ( - mention_type in user_mention_types - ) - is_chat_mention_signature = isinstance(mention_data, ChatMention) and ( - mention_type in chat_mention_types - ) - is_mention_all_signature = mention_type == MentionTypes.all_members - - if not any( # noqa: WPS337 - { - is_chat_mention_signature, - is_mention_all_signature, - is_user_mention_signature, - }, - ): - raise ValueError("No one suitable type for this mention_data signature") - - return mention_type - - @classmethod - def build_from_values( - cls, - mention_type: MentionTypes, - mentioned_entity_id: UUID, - name: Optional[str] = None, - mention_id: Optional[UUID] = None, - ) -> "Mention": - """Build mention. - - Simpler to use than constructor 'cause of flat values. - - Arguments: - mention_type: mention type. - mentioned_entity_id: id of mentioned entity (user, chat, etc.). - name: for overriding mention name. - mention_id: mention id (if not passed, will be generated). - - Raises: - NotImplementedError: If unsupported mention type was passed. - - Returns: - Built mention. - """ - mention_data: Union[UserMention, ChatMention] - - if mention_type in {MentionTypes.user, MentionTypes.contact}: - mention_data = UserMention(user_huid=mentioned_entity_id, name=name) - elif mention_type in {MentionTypes.chat, MentionTypes.channel}: - mention_data = ChatMention(group_chat_id=mentioned_entity_id, name=name) - else: - raise NotImplementedError("Unsupported mention type") - - return cls( - mention_id=mention_id, - mention_data=mention_data, - mention_type=mention_type, - ) - - def to_botx_format(self) -> str: - """Format mention to format, which can be parse by botx. - - Raises: - NotImplementedError: If unsupported mention type was passed. - - Returns: - Formatted mention. - """ - formatted_mention_data = "{{mention:{0}}}".format(self.mention_id) - - if self.mention_type == MentionTypes.user: - prefix = "@" - elif self.mention_type == MentionTypes.contact: - prefix = "@@" - elif self.mention_type in {MentionTypes.chat, MentionTypes.channel}: - prefix = "##" - else: - raise NotImplementedError("Unsupported mention type") - - return "{0}{1}".format(prefix, formatted_mention_data) - - -class Reply(BotXBaseModel): - """Message that was replied.""" - - #: attachment metadata. - attachment_meta: Optional[AttachmentMeta] = Field(alias="attachment") - - #: text of source message. - body: Optional[str] - - #: mentions of source message. - mentions: List[Mention] = [] - - #: type of source message's chat. - reply_type: ChatTypes - - #: uuid of sender. - sender: UUID - - #: chat name of source message. - source_chat_name: Optional[str] - - #: chat uuid of source message. - source_group_chat_id: Optional[UUID] - - #: uuid of source message. - source_sync_id: UUID - - class Config: - allow_population_by_field_name = True - - -class Entity(BotXBaseModel): - """Additional entity that can be received by bot.""" - - #: entity type. - type: EntityTypes # noqa: WPS125 - - #: entity data. - data: Union[Forward, Mention, Reply] # noqa: WPS110 - - -class EntityList(BotXBaseModel): - """Additional wrapped class for use property.""" - - __root__: List[Entity] - - @property - def mentions(self) -> List[Mention]: - """Search mentions in message's entity. - - Returns: - List of mentions. - """ - return [ - cast(Mention, entity.data) - for entity in self.__root__ - if entity.type == EntityTypes.mention - ] - - @property - def forward(self) -> Forward: - """Search forward in message's entity. - - Returns: - Information about forward. - - Raises: - AttributeError: raised if message has no forward. - """ - for entity in self.__root__: - if entity.type == EntityTypes.forward: # pragma: no branch - return cast(Forward, entity.data) - raise AttributeError("forward") - - @property - def reply(self) -> Reply: - """Search reply in message's entity. - - Returns: - Reply. - - Raises: - AttributeError: raised if message has no reply. - """ - for entity in self.__root__: - if entity.type == EntityTypes.reply: # pragma: no branch - return cast(Reply, entity.data) - raise AttributeError("reply") diff --git a/botx/models/enums.py b/botx/models/enums.py index f395b451..5e430071 100644 --- a/botx/models/enums.py +++ b/botx/models/enums.py @@ -1,168 +1,239 @@ -"""Definition of enums that are used across different components of this library.""" +from enum import Enum, auto -from enum import Enum +from botx.models.api_base import StrEnum -class SystemEvents(Enum): - """System enums that bot can retrieve from BotX API in message. +class AutoName(Enum): + def _generate_next_value_( # type: ignore # noqa: WPS120 + name, # noqa: N805 (copied from official python docs) + start, + count, + last_values, + ): + return name - !!! info - NOTE: `file_transfer` is not a system event, but it is logical to place it in - this enum. - """ - - #: `system:chat_created` event. - chat_created = "system:chat_created" - - #: `system:added_to_chat` event. - added_to_chat = "system:added_to_chat" - - #: `system:deleted_from_chat` event. - deleted_from_chat = "system:deleted_from_chat" - - #: `system:left_from_chat` event. - left_from_chat = "system:left_from_chat" - - #: `system:internal_bot_notification` event - internal_bot_notification = "system:internal_bot_notification" - - #: `system:cts_login` event. - cts_login = "system:cts_login" - - #: `system:cts_logout` event. - cts_logout = "system:cts_logout" - #: `system:smartapp_event` event. - smartapp_event = "system:smartapp_event" +class UserKinds(AutoName): + RTS_USER = auto() + CTS_USER = auto() + BOT = auto() - #: `file_transfer` message. - file_transfer = "file_transfer" +class AttachmentTypes(AutoName): + IMAGE = auto() + VIDEO = auto() + DOCUMENT = auto() + VOICE = auto() + LOCATION = auto() + CONTACT = auto() + LINK = auto() -class CommandTypes(str, Enum): - """Enum that specify from whom command was received.""" - #: command received from user. - user = "user" +class ClientPlatforms(AutoName): + WEB = auto() + ANDROID = auto() + IOS = auto() + DESKTOP = auto() - #: command received from system. - system = "system" +class MentionTypes(AutoName): + CONTACT = auto() + CHAT = auto() + CHANNEL = auto() + USER = auto() + ALL = auto() -class ChatTypes(str, Enum): - """Enum for type of chat.""" - #: private chat for user with bot. - chat = "chat" +class ChatTypes(AutoName): + """BotX chat types. - #: chat with several users. - group_chat = "group_chat" - - #: channel chat. - channel = "channel" - - # botx - botx = "botx" # todo replies incoming with whith type - - -class UserKinds(str, Enum): - """Enum for type of user.""" - - #: normal user. - user = "user" - - #: normal user, but will present if all users in chat are from the same CTS. - cts_user = "cts_user" - - #: bot user. - bot = "botx" + Attributes: + PERSONAL_CHAT: Personal chat with user. + GROUP_CHAT: Group chat. + CHANNEL: Public channel. + """ + PERSONAL_CHAT = auto() + GROUP_CHAT = auto() + CHANNEL = auto() -class Statuses(str, Enum): - """Enum for status of operation in BotX API.""" - #: operation was successfully proceed. - ok = "ok" +class APIChatTypes(StrEnum): + CHAT = "chat" + GROUP_CHAT = "group_chat" + CHANNEL = "channel" - #: there was an error while processing operation. - error = "error" +class BotAPICommandTypes(StrEnum): + USER = "user" + SYSTEM = "system" -class EntityTypes(str, Enum): - """Types for entities that could be received by bot.""" - #: mention entity. - mention = "mention" +class BotAPIClientPlatforms(StrEnum): + WEB = "web" + ANDROID = "android" + IOS = "ios" + DESKTOP = "desktop" - #: forward entity. - forward = "forward" - #: reply entity. - reply = "reply" +class BotAPIEntityTypes(StrEnum): + MENTION = "mention" + FORWARD = "forward" + REPLY = "reply" -class AttachmentsTypes(str, Enum): - """Types for attachments that could be received by bot.""" +class BotAPIMentionTypes(StrEnum): + CONTACT = "contact" + CHAT = "chat" + CHANNEL = "channel" + USER = "user" + ALL = "all" - image = "image" - video = "video" - document = "document" - voice = "voice" - contact = "contact" - location = "location" - link = "link" +class APIUserKinds(StrEnum): + USER = "user" + CTS_USER = "cts_user" + BOTX = "botx" -class MentionTypes(str, Enum): - """Enum for available validated_values in mentions.""" - #: mention single user from chat in message. - user = "user" +class APIAttachmentTypes(StrEnum): + IMAGE = "image" + VIDEO = "video" + DOCUMENT = "document" + VOICE = "voice" + LOCATION = "location" + CONTACT = "contact" + LINK = "link" - #: mention user by user_huid. - contact = "contact" - #: mention chat in message. - chat = "chat" +def convert_client_platform_to_domain( + client_platform: BotAPIClientPlatforms, +) -> ClientPlatforms: + client_platforms_mapping = { + BotAPIClientPlatforms.WEB: ClientPlatforms.WEB, + BotAPIClientPlatforms.ANDROID: ClientPlatforms.ANDROID, + BotAPIClientPlatforms.IOS: ClientPlatforms.IOS, + BotAPIClientPlatforms.DESKTOP: ClientPlatforms.DESKTOP, + } - #: mention channel in message. - channel = "channel" + converted_type = client_platforms_mapping.get(client_platform) + if converted_type is None: + raise NotImplementedError(f"Unsupported client platform: {client_platform}") - #: mention all users in chat - all_members = "all" + return converted_type -class LinkProtos(str, Enum): - """Enum for protos of links in attachments.""" +def convert_mention_type_to_domain(mention_type: BotAPIMentionTypes) -> MentionTypes: + mention_types_mapping = { + BotAPIMentionTypes.CONTACT: MentionTypes.CONTACT, + BotAPIMentionTypes.CHAT: MentionTypes.CHAT, + BotAPIMentionTypes.CHANNEL: MentionTypes.CHANNEL, + BotAPIMentionTypes.USER: MentionTypes.USER, + BotAPIMentionTypes.ALL: MentionTypes.ALL, + } - #: proto for attach with email. - email = "mailto:" + converted_type = mention_types_mapping.get(mention_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported mention type: {mention_type}") - #: proto for attach with telephone number. - telephone = "tel://" + return converted_type -class ClientPlatformEnum(str, Enum): - """Enum for distinguishing client platforms.""" +def convert_mention_type_from_domain( + mention_type: MentionTypes, +) -> BotAPIMentionTypes: + embed_mention_types_mapping = { + MentionTypes.USER: BotAPIMentionTypes.USER, + MentionTypes.CONTACT: BotAPIMentionTypes.CONTACT, + MentionTypes.CHAT: BotAPIMentionTypes.CHAT, + MentionTypes.CHANNEL: BotAPIMentionTypes.CHANNEL, + MentionTypes.ALL: BotAPIMentionTypes.ALL, + } - #: Web platform. - web = "web" + converted_type = embed_mention_types_mapping.get(mention_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported mention type: {mention_type}") - #: Android platform. - android = "android" + return converted_type - #: iOS platform. - ios = "ios" - #: Desktop platform. - desktop = "desktop" +def convert_user_kind_to_domain(user_kind: APIUserKinds) -> UserKinds: + user_kinds_mapping = { + APIUserKinds.USER: UserKinds.RTS_USER, + APIUserKinds.CTS_USER: UserKinds.CTS_USER, + APIUserKinds.BOTX: UserKinds.BOT, + } + converted_type = user_kinds_mapping.get(user_kind) + if converted_type is None: + raise NotImplementedError(f"Unsupported user kind: {user_kind}") -class ButtonHandlerTypes(str, Enum): - """Enum for markup's `handler` field.""" + return converted_type + + +def convert_attachment_type_to_domain( + attachment_type: APIAttachmentTypes, +) -> AttachmentTypes: + attachment_types_mapping = { + APIAttachmentTypes.IMAGE: AttachmentTypes.IMAGE, + APIAttachmentTypes.VIDEO: AttachmentTypes.VIDEO, + APIAttachmentTypes.DOCUMENT: AttachmentTypes.DOCUMENT, + APIAttachmentTypes.VOICE: AttachmentTypes.VOICE, + APIAttachmentTypes.LOCATION: AttachmentTypes.LOCATION, + APIAttachmentTypes.CONTACT: AttachmentTypes.CONTACT, + APIAttachmentTypes.LINK: AttachmentTypes.LINK, + } + + converted_type = attachment_types_mapping.get(attachment_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported attachment type: {attachment_type}") + + return converted_type + + +def convert_attachment_type_from_domain( + attachment_type: AttachmentTypes, +) -> APIAttachmentTypes: + attachment_types_mapping = { + AttachmentTypes.IMAGE: APIAttachmentTypes.IMAGE, + AttachmentTypes.VIDEO: APIAttachmentTypes.VIDEO, + AttachmentTypes.DOCUMENT: APIAttachmentTypes.DOCUMENT, + AttachmentTypes.VOICE: APIAttachmentTypes.VOICE, + AttachmentTypes.LOCATION: APIAttachmentTypes.LOCATION, + AttachmentTypes.CONTACT: APIAttachmentTypes.CONTACT, + AttachmentTypes.LINK: APIAttachmentTypes.LINK, + } + + converted_type = attachment_types_mapping.get(attachment_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported attachment type: {attachment_type}") + + return converted_type + + +def convert_chat_type_from_domain(chat_type: ChatTypes) -> APIChatTypes: + chat_types_mapping = { + ChatTypes.PERSONAL_CHAT: APIChatTypes.CHAT, + ChatTypes.GROUP_CHAT: APIChatTypes.GROUP_CHAT, + ChatTypes.CHANNEL: APIChatTypes.CHANNEL, + } + + converted_type = chat_types_mapping.get(chat_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported chat type: {chat_type}") + + return converted_type + + +def convert_chat_type_to_domain(chat_type: APIChatTypes) -> ChatTypes: + chat_types_mapping = { + APIChatTypes.CHAT: ChatTypes.PERSONAL_CHAT, + APIChatTypes.GROUP_CHAT: ChatTypes.GROUP_CHAT, + APIChatTypes.CHANNEL: ChatTypes.CHANNEL, + } - #: bot side process. - bot = "bot" + converted_type = chat_types_mapping.get(chat_type) + if converted_type is None: + raise NotImplementedError(f"Unsupported chat type: {chat_type}") - #: client side process. - client = "client" + return converted_type diff --git a/botx/models/errors.py b/botx/models/errors.py deleted file mode 100644 index d9b259b1..00000000 --- a/botx/models/errors.py +++ /dev/null @@ -1,24 +0,0 @@ -"""Definition of errors in processing request from BotX API.""" - -from typing import List - -from botx.models.base import BotXBaseModel - - -class BotDisabledErrorData(BotXBaseModel): - """Data about occurred error.""" - - #: message that will be shown to user. - status_message: str - - -class BotDisabledResponse(BotXBaseModel): - """Response to BotX API if there was an error in handling incoming request.""" - - #: error reason. *This should always be `bot_disabled` string.* - reason: str = "bot_disabled" - - #: data about occurred error that should include `status_message` field in json. - error_data: BotDisabledErrorData - - errors: List[str] = [] diff --git a/botx/models/events.py b/botx/models/events.py deleted file mode 100644 index 4234508a..00000000 --- a/botx/models/events.py +++ /dev/null @@ -1,114 +0,0 @@ -"""Definition of different schemas for system events.""" - -from types import MappingProxyType -from typing import Any, Dict, List, Mapping, Optional, Type -from uuid import UUID - -from botx.clients.types.message_payload import InternalBotNotificationPayload -from botx.models.base import BotXBaseModel -from botx.models.enums import ChatTypes, SystemEvents -from botx.models.users import UserInChatCreated - - -class ChatCreatedEvent(BotXBaseModel): - """Shape for `system:chat_created` event data.""" - - #: chat id from which event received. - group_chat_id: UUID - - #: type of chat. - chat_type: ChatTypes - - #: chat name. - name: str - - #: HUID of user that created chat. - creator: UUID - - #: list of users that are members of chat. - members: List[UserInChatCreated] - - -class AddedToChatEvent(BotXBaseModel): - """Shape for `system:added_to_chat` event data.""" - - #: members added to chat. - added_members: List[UUID] - - -class DeletedFromChatEvent(BotXBaseModel): - """Shape for `system:deleted_from_chat` event data.""" - - #: members deleted from chat - deleted_members: List[UUID] - - -class LeftFromChatEvent(BotXBaseModel): - """Shape for `system:left_from_chat` event data.""" - - #: left chat members - left_members: List[UUID] - - -class InternalBotNotificationEvent(BotXBaseModel): - """Shape for `system:internal_bot_notification` event data.""" - - #: notification data - data: InternalBotNotificationPayload # noqa: WPS110 - - #: user-defined extra options - opts: Dict[str, Any] - - -class CTSLoginEvent(BotXBaseModel): - """Shape for `system:cts_login` event data.""" - - #: huid of user which logged into CTS. - user_huid: UUID - - #: CTS id. - cts_id: UUID - - -class CTSLogoutEvent(BotXBaseModel): - """Shape for `system:cts_logout` event data.""" - - #: huid of user which logged out from CTS. - user_huid: UUID - - #: CTS id. - cts_id: UUID - - -class SmartAppEvent(BotXBaseModel): - """Shape for `system:smartapp_event` event data.""" - - #: unique request id - ref: Optional[UUID] = None - - #: smartapp id - smartapp_id: UUID - - #: event data - data: Dict[str, Any] # noqa: WPS110 - - #: event options - opts: Dict[str, Any] = {} - - #: version of protocol smartapp <-> bot - smartapp_api_version: int - - -# dict for validating shape for different events -EVENTS_SHAPE_MAP: Mapping[SystemEvents, Type[BotXBaseModel]] = MappingProxyType( - { - SystemEvents.chat_created: ChatCreatedEvent, - SystemEvents.added_to_chat: AddedToChatEvent, - SystemEvents.deleted_from_chat: DeletedFromChatEvent, - SystemEvents.left_from_chat: LeftFromChatEvent, - SystemEvents.internal_bot_notification: InternalBotNotificationEvent, - SystemEvents.cts_login: CTSLoginEvent, - SystemEvents.cts_logout: CTSLogoutEvent, - SystemEvents.smartapp_event: SmartAppEvent, - }, -) diff --git a/botx/models/files.py b/botx/models/files.py deleted file mode 100644 index 60912bd3..00000000 --- a/botx/models/files.py +++ /dev/null @@ -1,352 +0,0 @@ -"""Definition of file that can be included in incoming message or in sending result.""" - -import base64 -from contextlib import contextmanager -from io import BytesIO -from pathlib import Path -from types import MappingProxyType -from typing import AnyStr, AsyncIterable, BinaryIO, Generator, Optional, TextIO, Union -from uuid import UUID - -from base64io import Base64IO - -from botx.models.base import BotXBaseModel -from botx.models.enums import AttachmentsTypes - -EXTENSIONS_TO_MIMETYPES = MappingProxyType( - { - # application - ".7z": "application/x-7z-compressed", - ".abw": "application/x-abiword", - ".ai": "application/postscript", - ".arc": "application/x-freearc", - ".azw": "application/vnd.amazon.ebook", - ".bin": "application/octet-stream", - ".bz": "application/x-bzip", - ".bz2": "application/x-bzip2", - ".cda": "application/x-cdf", - ".csh": "application/x-csh", - ".doc": "application/msword", - ".docx": ( - "application/vnd.openxmlformats-officedocument.wordprocessingml.document" - ), - ".eot": "application/vnd.ms-fontobject", - ".eps": "application/postscript", - ".epub": "application/epub+zip", - ".gz": "application/gzip", - ".jar": "application/java-archive", - ".json-api": "application/vnd.api+json", - ".json-patch": "application/json-patch+json", - ".json": "application/json", - ".jsonld": "application/ld+json", - ".mdb": "application/x-msaccess", - ".mpkg": "application/vnd.apple.installer+xml", - ".odp": "application/vnd.oasis.opendocument.presentation", - ".ods": "application/vnd.oasis.opendocument.spreadsheet", - ".odt": "application/vnd.oasis.opendocument.text", - ".ogx": "application/ogg", - ".pdf": "application/pdf", - ".php": "application/x-httpd-php", - ".ppt": "application/vnd.ms-powerpoint", - ".pptx": ( - "application/vnd.openxmlformats-officedocument.presentationml.presentation" - ), - ".ps": "application/postscript", - ".rar": "application/vnd.rar", - ".rtf": "application/rtf", - ".sh": "application/x-sh", - ".swf": "application/x-shockwave-flash", - ".tar": "application/x-tar", - ".vsd": "application/vnd.visio", - ".wasm": "application/wasm", - ".webmanifest": "application/manifest+json", - ".xhtml": "application/xhtml+xml", - ".xls": "application/vnd.ms-excel", - ".xlsx": "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet", - ".xul": "application/vnd.mozilla.xul+xml", - ".zip": "application/zip", - # audio - ".aac": "audio/aac", - ".mid": "audio/midi", - ".midi": "audio/midi", - ".mp3": "audio/mpeg", - ".oga": "audio/ogg", - ".opus": "audio/opus", - ".wav": "audio/wav", - ".weba": "audio/webm", - # font - ".otf": "font/otf", - ".ttf": "font/ttf", - ".woff": "font/woff", - ".woff2": "font/woff2", - # image - ".avif": "image/avif", - ".bmp": "image/bmp", - ".gif": "image/gif", - ".ico": "image/vnd.microsoft.icon", - ".jpeg": "image/jpeg", - ".jpg": "image/jpeg", - ".png": "image/png", - ".svg": "image/svg+xml", - ".svgz": "image/svg+xml", - ".tif": "image/tiff", - ".tiff": "image/tiff", - ".webp": "image/webp", - # text - ".css": "text/css", - ".csv": "text/csv", - ".htm": "text/html", - ".html": "text/html", - ".ics": "text/calendar", - ".js": "text/javascript", - ".mjs": "text/javascript", - ".txt": "text/plain", - ".text": "text/plain", - ".xml": "text/xml", - # video - ".3g2": "video/3gpp2", - ".3gp": "video/3gpp", - ".avi": "video/x-msvideo", - ".mov": "video/quicktime", - ".mp4": "video/mp4", - ".mpeg": "video/mpeg", - ".mpg": "video/mpeg", - ".ogv": "video/ogg", - ".ts": "video/mp2t", - ".webm": "video/webm", - ".wmv": "video/x-ms-wmv", - }, -) -DEFAULT_MIMETYPE = "application/octet-stream" - - -class NamedAsyncIterable(AsyncIterable): - """AsyncIterable with `name` protocol.""" - - name: str - - -class File(BotXBaseModel): # noqa: WPS214 - """Object that represents file in RFC 2397 format.""" - - #: name of file. - file_name: str - - #: file content in RFC 2397 format. - data: str # noqa: WPS110 - - #: text under file. - caption: Optional[str] = None - - @classmethod - def from_file( # noqa: WPS210 - cls, - file: Union[TextIO, BinaryIO], - filename: Optional[str] = None, - ) -> "File": - """Convert file-like object into BotX API compatible file. - - Arguments: - file: file-like object that will be used for creating file. - filename: name that will be used for file, if was not passed, then will be - retrieved from `file` `.name` property. - - Returns: - Built file object. - """ - filename = filename or Path(file.name).name - encoded_file = BytesIO() - - text_mode = file.read(0) == "" # b"" if bytes mode - - with Base64IO(encoded_file) as b64_stream: - if text_mode: - for text_line in file: # TODO: Deprecate text mode in 0.17 - b64_stream.write(text_line.encode()) # type: ignore - else: - for line in file: - b64_stream.write(line) - - encoded_file.seek(0) - encoded_data = encoded_file.read().decode() - - media_type = cls._get_mimetype(filename) - return cls(file_name=filename, data=cls._to_rfc2397(media_type, encoded_data)) - - @classmethod - async def async_from_file( # noqa: WPS210 - cls, - file: NamedAsyncIterable, - filename: Optional[str] = None, - ) -> "File": - """Convert async file-like object into BotX API compatible file. - - Arguments: - file: async file-like object that will be used for creating file. - filename: name that will be used for file, if was not passed, then will be - retrieved from `file` `.name` property. - - Returns: - Built File object. - """ - assert hasattr( # noqa: WPS421 - file, - "__aiter__", - ), "file should support async iteration" - - filename = filename or Path(file.name).name - media_type = cls._get_mimetype(filename) - - encoded_file = BytesIO() - - with Base64IO(encoded_file) as b64_stream: - async for line in file: # pragma: no branch - b64_stream.write(line) - - encoded_file.seek(0) - encoded_data = encoded_file.read().decode() - - return cls(file_name=filename, data=cls._to_rfc2397(media_type, encoded_data)) - - @contextmanager - def file_chunks(self) -> Generator[bytes, None, None]: - """Return file data in iterator that will return bytes.""" - encoded_file = BytesIO(self.data_in_base64.encode()) - - with Base64IO(encoded_file) as decoded_file: - yield decoded_file - - @classmethod - def from_string(cls, data_of_file: AnyStr, filename: str) -> "File": - """Build file from bytes or string passed to method in `data` with `filename` as name. - - Arguments: - data_of_file: string or bytes that will be used for creating file. - filename: name for new file. - - Returns: - Built file object. - """ - if isinstance(data_of_file, str): - file_data = data_of_file.encode() - else: - file_data = data_of_file - file = BytesIO(file_data) - file.name = filename - return cls.from_file(file) - - @property - def file(self) -> BinaryIO: - """Return file data in file-like object that will return bytes.""" - bytes_file = BytesIO(self.data_in_bytes) - bytes_file.name = self.file_name - return bytes_file - - @property - def size_in_bytes(self) -> int: - """Return file size in bytes.""" - with self.file_chunks() as chunks: - return sum(len(chunk) for chunk in chunks) # type: ignore - - @property - def data_in_bytes(self) -> bytes: - """Return decoded file data in bytes.""" - return base64.b64decode(self.data_in_base64) - - @property - def data_in_base64(self) -> str: - """Return file data in base64 encoded string.""" - return self.data.split(",", 1)[1] - - @property - def media_type(self) -> str: - """Return media type of file.""" - return self._get_mimetype(self.file_name) - - @classmethod - def get_ext_by_mimetype(cls, mimetype: str) -> Optional[str]: - """Get extension by mimetype. - - Arguments: - mimetype: mimetype of file. - - Returns: - file extension or none if mimetype not found. - """ - for ext, m_type in EXTENSIONS_TO_MIMETYPES.items(): - if m_type == mimetype: - return ext - - return None - - @classmethod - def _to_rfc2397(cls, media_type: str, encoded_data: str) -> str: - """Apply RFC 2397 format to encoded file contents. - - Arguments: - media_type: file media type. - encoded_data: base64 encoded file contents. - - Returns: - File contents converted to RFC 2397. - """ - return "data:{0};base64,{1}".format(media_type, encoded_data) - - @classmethod - def _get_mimetype(cls, filename: str) -> str: - """Get mimetype by filename. - - Arguments: - filename: file name to inspect. - - Returns: - File mimetype. - """ - file_extension = Path(filename).suffix.lower() - return EXTENSIONS_TO_MIMETYPES.get(file_extension, DEFAULT_MIMETYPE) - - -class MetaFile(BotXBaseModel): - """File info from file service.""" - - #: type of file - type: AttachmentsTypes - - #: file url. - file: str - - #: mime type of file. - file_mime_type: str - - #: name of file. - file_name: str - - #: file preview. - file_preview: Optional[str] - - #: height of file (px). - file_preview_height: Optional[int] - - #: width of file (px). - file_preview_width: Optional[int] - - #: size of file. - file_size: int - - #: hash of file. - file_hash: str - - #: encryption algorithm of file. - file_encryption_algo: str - - #: chunks size. - chunk_size: int - - #: ID of file. - file_id: UUID - - #: file caption. - caption: Optional[str] - - #: media file duration. - duration: Optional[int] diff --git a/botx/models/menu.py b/botx/models/menu.py deleted file mode 100644 index 931ee6ea..00000000 --- a/botx/models/menu.py +++ /dev/null @@ -1,42 +0,0 @@ -"""Pydantic models for bot menu.""" - -from typing import List - -from botx.models.base import BotXBaseModel -from botx.models.enums import Statuses - - -class MenuCommand(BotXBaseModel): - """Command that is shown in bot menu.""" - - #: command description that will be shown in menu. - description: str - - #: command body that will trigger command execution. - body: str - - #: command name. - name: str - - -class StatusResult(BotXBaseModel): - """Bot menu commands collection.""" - - #: is bot enabled. - enabled: bool = True - - #: status of bot. - status_message: str = "Bot is working" - - #: list of bot commands that will be shown in menu. - commands: List[MenuCommand] = [] - - -class Status(BotXBaseModel): - """Object that should be returned on `/status` request from BotX API.""" - - #: operation status. - status: Statuses = Statuses.ok - - #: bot status. - result: StatusResult = StatusResult() diff --git a/tests/test_bots/__init__.py b/botx/models/message/__init__.py similarity index 100% rename from tests/test_bots/__init__.py rename to botx/models/message/__init__.py diff --git a/botx/models/message/edit_message.py b/botx/models/message/edit_message.py new file mode 100644 index 00000000..11b0e0eb --- /dev/null +++ b/botx/models/message/edit_message.py @@ -0,0 +1,19 @@ +from dataclasses import dataclass +from typing import Any, Dict, Union +from uuid import UUID + +from botx.missing import Missing, Undefined +from botx.models.attachments import IncomingFileAttachment, OutgoingAttachment +from botx.models.message.markup import BubbleMarkup, KeyboardMarkup + + +@dataclass +class EditMessage: + bot_id: UUID + sync_id: UUID + body: Missing[str] = Undefined + metadata: Missing[Dict[str, Any]] = Undefined + bubbles: Missing[BubbleMarkup] = Undefined + keyboard: Missing[KeyboardMarkup] = Undefined + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined + markup_auto_adjust: Missing[bool] = Undefined diff --git a/botx/models/message/forward.py b/botx/models/message/forward.py new file mode 100644 index 00000000..a8d02c32 --- /dev/null +++ b/botx/models/message/forward.py @@ -0,0 +1,24 @@ +from dataclasses import dataclass +from typing import Literal +from uuid import UUID + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.enums import BotAPIEntityTypes + + +@dataclass +class Forward: + chat_id: UUID + author_id: UUID + sync_id: UUID + + +class BotAPIForwardData(VerifiedPayloadBaseModel): + group_chat_id: UUID + sender_huid: UUID + source_sync_id: UUID + + +class BotAPIForward(VerifiedPayloadBaseModel): + type: Literal[BotAPIEntityTypes.FORWARD] + data: BotAPIForwardData diff --git a/botx/models/message/incoming_message.py b/botx/models/message/incoming_message.py new file mode 100644 index 00000000..a771458c --- /dev/null +++ b/botx/models/message/incoming_message.py @@ -0,0 +1,281 @@ +from dataclasses import dataclass, field +from types import SimpleNamespace +from typing import Any, Dict, List, Optional, Tuple, Union, cast +from uuid import UUID + +from pydantic import Field + +from botx.models.async_files import APIAsyncFile, File, convert_async_file_to_domain +from botx.models.attachments import ( + AttachmentContact, + AttachmentLink, + AttachmentLocation, + BotAPIAttachment, + FileAttachmentBase, + IncomingFileAttachment, + convert_api_attachment_to_domain, +) +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotAPICommandPayload, + BotAPIDeviceContext, + BotAPIUserContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import ( + AttachmentTypes, + BotAPIEntityTypes, + BotAPIMentionTypes, + ClientPlatforms, + convert_chat_type_to_domain, + convert_client_platform_to_domain, + convert_mention_type_to_domain, +) +from botx.models.message.forward import BotAPIForward, Forward +from botx.models.message.mentions import ( + BotAPIMention, + BotAPIMentionData, + BotAPINestedMentionData, + Mention, + MentionList, +) +from botx.models.message.reply import BotAPIReply, Reply + + +@dataclass +class UserDevice: + manufacturer: Optional[str] + device_name: Optional[str] + os: Optional[str] + pushes: Optional[bool] + timezone: Optional[str] + permissions: Optional[Dict[str, Any]] + platform: Optional[ClientPlatforms] + platform_package_id: Optional[str] + app_version: Optional[str] + locale: Optional[str] + + +@dataclass +class UserSender: + huid: UUID + ad_login: Optional[str] + ad_domain: Optional[str] + username: Optional[str] + is_chat_admin: Optional[bool] + is_chat_creator: Optional[bool] + device: UserDevice + + @property + def upn(self) -> Optional[str]: + # https://docs.microsoft.com/en-us/windows/win32/secauthn/user-name-formats + if not (self.ad_login and self.ad_domain): + return None + + return f"{self.ad_login}@{self.ad_domain}" + + +@dataclass +class IncomingMessage(BotCommandBase): + sync_id: UUID + source_sync_id: Optional[UUID] + body: str + data: Dict[str, Any] + metadata: Dict[str, Any] + sender: UserSender + chat: Chat + mentions: MentionList = field(default_factory=MentionList) + forward: Optional[Forward] = None + reply: Optional[Reply] = None + file: Optional[Union[File, IncomingFileAttachment]] = None + location: Optional[AttachmentLocation] = None + contact: Optional[AttachmentContact] = None + link: Optional[AttachmentLink] = None + + state: SimpleNamespace = field(default_factory=SimpleNamespace) + + @property + def argument(self) -> str: + split_body = self.body.split() + if not split_body: + return "" + + command_len = len(split_body[0]) + return self.body[command_len:].strip() + + @property + def arguments(self) -> Tuple[str, ...]: + return tuple(arg.strip() for arg in self.argument.split()) + + +BotAPIEntity = Union[BotAPIMention, BotAPIForward, BotAPIReply] +Entity = Union[Mention, Forward, Reply] + + +def _convert_bot_api_mention_to_domain(api_mention_data: BotAPIMentionData) -> Mention: + entity_id: Optional[UUID] = None + name: Optional[str] = None + + if api_mention_data.mention_type != BotAPIMentionTypes.ALL: + mention_data = cast(BotAPINestedMentionData, api_mention_data.mention_data) + entity_id = mention_data.entity_id + name = mention_data.name + + return Mention( + type=convert_mention_type_to_domain(api_mention_data.mention_type), + entity_id=entity_id, + name=name, + ) + + +def convert_bot_api_entity_to_domain(api_entity: BotAPIEntity) -> Entity: + if api_entity.type == BotAPIEntityTypes.MENTION: + api_entity = cast(BotAPIMention, api_entity) + return _convert_bot_api_mention_to_domain(api_entity.data) + + if api_entity.type == BotAPIEntityTypes.FORWARD: + api_entity = cast(BotAPIForward, api_entity) + + return Forward( + chat_id=api_entity.data.group_chat_id, + author_id=api_entity.data.sender_huid, + sync_id=api_entity.data.source_sync_id, + ) + + if api_entity.type == BotAPIEntityTypes.REPLY: + api_entity = cast(BotAPIReply, api_entity) + + mentions = MentionList() + for api_mention_data in api_entity.data.mentions: + mentions.append(_convert_bot_api_mention_to_domain(api_mention_data)) + + return Reply( + author_id=api_entity.data.sender, + sync_id=api_entity.data.source_sync_id, + body=api_entity.data.body, + mentions=mentions, + ) + + raise NotImplementedError(f"Unsupported entity type: {api_entity.type}") + + +class BotAPIIncomingMessageContext( + BotAPIUserContext, + BotAPIChatContext, + BotAPIDeviceContext, +): + """Class for merging contexts.""" + + +class BotAPIIncomingMessage(BotAPIBaseCommand): + payload: BotAPICommandPayload = Field(..., alias="command") + sender: BotAPIIncomingMessageContext = Field(..., alias="from") + + source_sync_id: Optional[UUID] + attachments: List[BotAPIAttachment] + async_files: List[APIAsyncFile] + entities: List[BotAPIEntity] + + def to_domain(self, raw_command: Dict[str, Any]) -> IncomingMessage: # noqa: WPS231 + if self.sender.device_meta: + pushes = self.sender.device_meta.pushes + timezone = self.sender.device_meta.timezone + permissions = self.sender.device_meta.permissions + else: + pushes, timezone, permissions = None, None, None + + device = UserDevice( + manufacturer=self.sender.manufacturer, + device_name=self.sender.device, + os=self.sender.device_software, + pushes=pushes, + timezone=timezone, + permissions=permissions, + platform=( + convert_client_platform_to_domain(self.sender.platform) + if self.sender.platform + else None + ), + platform_package_id=self.sender.platform_package_id, + app_version=self.sender.app_version, + locale=self.sender.locale, + ) + + sender = UserSender( + huid=self.sender.user_huid, + ad_login=self.sender.ad_login, + ad_domain=self.sender.ad_domain, + username=self.sender.username, + is_chat_admin=self.sender.is_admin, + is_chat_creator=self.sender.is_creator, + device=device, + ) + + chat = Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ) + + file: Optional[Union[File, IncomingFileAttachment]] = None + location: Optional[AttachmentLocation] = None + contact: Optional[AttachmentContact] = None + link: Optional[AttachmentLink] = None + if self.async_files: + # Always one async file per-message + file = convert_async_file_to_domain(self.async_files[0]) + elif self.attachments: + # Always one attachment per-message + attachment_domain = convert_api_attachment_to_domain(self.attachments[0]) + if isinstance(attachment_domain, FileAttachmentBase): + file = attachment_domain + elif attachment_domain.type == AttachmentTypes.LOCATION: + location = attachment_domain + elif attachment_domain.type == AttachmentTypes.CONTACT: + contact = attachment_domain + elif attachment_domain.type == AttachmentTypes.LINK: + link = attachment_domain + else: + raise NotImplementedError + + mentions: MentionList = MentionList() + forward: Optional[Forward] = None + reply: Optional[Reply] = None + for entity in self.entities: + entity_domain = convert_bot_api_entity_to_domain(entity) + if isinstance(entity_domain, Mention): + mentions.append(entity_domain) + elif isinstance(entity_domain, Forward): + # Max one forward per message + forward = entity_domain + elif isinstance(entity_domain, Reply): + # Max one reply per message + reply = entity_domain + else: + raise NotImplementedError + + bot = BotAccount( + id=self.bot_id, + host=self.sender.host, + ) + + return IncomingMessage( + bot=bot, + sync_id=self.sync_id, + source_sync_id=self.source_sync_id, + body=self.payload.body, + data=self.payload.data, + metadata=self.payload.metadata, + sender=sender, + chat=chat, + raw_command=raw_command, + file=file, + location=location, + contact=contact, + link=link, + mentions=mentions, + forward=forward, + reply=reply, + ) diff --git a/botx/models/message/markup.py b/botx/models/message/markup.py new file mode 100644 index 00000000..0f26b0ab --- /dev/null +++ b/botx/models/message/markup.py @@ -0,0 +1,131 @@ +from dataclasses import dataclass, field +from typing import Any, Dict, Iterator, List, Literal, Optional, Union + +from botx.missing import Missing, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel + + +@dataclass +class Button: + command: str + label: str + data: Dict[str, Any] = field(default_factory=dict) + + silent: bool = True # BotX has `False` as default, so Missing type can't be used + width_ratio: Missing[int] = Undefined + alert: Missing[str] = Undefined + process_on_client: Missing[bool] = Undefined + + +ButtonRow = List[Button] + + +class BaseMarkup: + def __init__(self, buttons: Optional[List[ButtonRow]] = None) -> None: + self._buttons = buttons or [] + + def __iter__(self) -> Iterator[ButtonRow]: + return iter(self._buttons) + + def __eq__(self, other: object) -> bool: + if not isinstance(other, BaseMarkup): + raise NotImplementedError + + # https://github.com/wemake-services/wemake-python-styleguide/issues/2172 + return self._buttons == other._buttons # noqa: WPS437 + + def add_built_button(self, button: Button, new_row: bool = True) -> None: + if new_row: + self._buttons.append([button]) + return + + if not self._buttons: + self._buttons.append([]) + + self._buttons[-1].append(button) + + def add_button( + self, + command: str, + label: str, + data: Optional[Dict[str, Any]] = None, + silent: bool = True, + width_ratio: Missing[int] = Undefined, + alert: Missing[str] = Undefined, + process_on_client: Missing[bool] = Undefined, + new_row: bool = True, + ) -> None: + button = Button( + command=command, + label=label, + data=data or {}, + silent=silent, + width_ratio=width_ratio, + alert=alert, + process_on_client=process_on_client, + ) + self.add_built_button(button, new_row=new_row) + + def add_row(self, button_row: ButtonRow) -> None: + self._buttons.append(button_row) + + +class BubbleMarkup(BaseMarkup): + """Class for managing inline message buttons.""" + + +class KeyboardMarkup(BaseMarkup): + """Class for managing keyboard message buttons.""" + + +Markup = Union[BubbleMarkup, KeyboardMarkup] + + +class BotXAPIButtonOptions(UnverifiedPayloadBaseModel): + silent: Missing[bool] + h_size: Missing[int] + show_alert: Missing[Literal[True]] + alert_text: Missing[str] + handler: Missing[Literal["client"]] + + +class BotXAPIButton(UnverifiedPayloadBaseModel): + command: str + label: str + data: Dict[str, Any] + opts: BotXAPIButtonOptions + + +class BotXAPIMarkup(UnverifiedPayloadBaseModel): + __root__: List[List[BotXAPIButton]] + + +def api_button_from_domain(button: Button) -> BotXAPIButton: + show_alert: Missing[Literal[True]] = Undefined + if button.alert is not Undefined: + show_alert = True + + handler: Missing[Literal["client"]] = Undefined + if button.process_on_client: + handler = "client" + + return BotXAPIButton( + command=button.command, + label=button.label, + data=button.data, + opts=BotXAPIButtonOptions( + silent=button.silent, + h_size=button.width_ratio, + alert_text=button.alert, + show_alert=show_alert, + handler=handler, + ), + ) + + +def api_markup_from_domain(markup: Markup) -> BotXAPIMarkup: + return BotXAPIMarkup( + __root__=[ + [api_button_from_domain(button) for button in buttons] for buttons in markup + ], + ) diff --git a/botx/models/message/mentions.py b/botx/models/message/mentions.py new file mode 100644 index 00000000..a254c8a4 --- /dev/null +++ b/botx/models/message/mentions.py @@ -0,0 +1,276 @@ +import re +from dataclasses import dataclass +from typing import Dict, List, Literal, Optional, Tuple, Union +from uuid import UUID, uuid4 + +from pydantic import Field, validator + +from botx.missing import Missing, Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel +from botx.models.enums import ( + BotAPIEntityTypes, + BotAPIMentionTypes, + MentionTypes, + convert_mention_type_from_domain, +) + + +@dataclass +class Mention: + type: MentionTypes + entity_id: Optional[UUID] = None + name: Optional[str] = None + + def __str__(self) -> str: + name = self.name or "" + entity_id = self.entity_id or "" + mention_type = self.type.value + return f"{mention_type}:{entity_id}:{name}" + + @classmethod + def user(cls, huid: UUID, name: Optional[str] = None) -> "Mention": + return cls( + type=MentionTypes.USER, + entity_id=huid, + name=name, + ) + + @classmethod + def contact(cls, huid: UUID, name: Optional[str] = None) -> "Mention": + return cls( + type=MentionTypes.CONTACT, + entity_id=huid, + name=name, + ) + + @classmethod + def chat(cls, chat_id: UUID, name: Optional[str] = None) -> "Mention": + return cls( + type=MentionTypes.CHAT, + entity_id=chat_id, + name=name, + ) + + @classmethod + def channel(cls, chat_id: UUID, name: Optional[str] = None) -> "Mention": + return cls( + type=MentionTypes.CHANNEL, + entity_id=chat_id, + name=name, + ) + + @classmethod + def all(cls) -> "Mention": + return cls(type=MentionTypes.ALL) + + +class MentionList(List[Mention]): + @property + def contacts(self) -> List[Mention]: + return [mention for mention in self if mention.type == MentionTypes.CONTACT] + + @property + def chats(self) -> List[Mention]: + return [mention for mention in self if mention.type == MentionTypes.CHAT] + + @property + def channels(self) -> List[Mention]: + return [mention for mention in self if mention.type == MentionTypes.CHANNEL] + + @property + def users(self) -> List[Mention]: + return [mention for mention in self if mention.type == MentionTypes.USER] + + @property + def all_users_mentioned(self) -> bool: + for mention in self: + if mention.type == MentionTypes.ALL: + return True + + return False + + +class BotAPINestedPersonalMentionData(VerifiedPayloadBaseModel): + entity_id: UUID = Field(alias="user_huid") + name: str + conn_type: str + + +class BotAPINestedGroupMentionData(VerifiedPayloadBaseModel): + entity_id: UUID = Field(alias="group_chat_id") + name: str + + +BotAPINestedMentionData = Union[ + BotAPINestedPersonalMentionData, + BotAPINestedGroupMentionData, +] + + +class BotAPIMentionData(VerifiedPayloadBaseModel): + mention_type: BotAPIMentionTypes + mention_id: UUID + mention_data: Optional[BotAPINestedMentionData] + + @validator("mention_data", pre=True) + @classmethod + def validate_mention_data( + cls, + mention_data: Dict[str, str], + ) -> Optional[Dict[str, str]]: + # Mention data can be an empty dict + if not mention_data: + return None + + return mention_data + + +class BotAPIMention(VerifiedPayloadBaseModel): + type: Literal[BotAPIEntityTypes.MENTION] + data: BotAPIMentionData + + +class BotXAPIPersonalMentionData(UnverifiedPayloadBaseModel): + user_huid: UUID + name: Missing[str] + + +class BotXAPIUserMention(UnverifiedPayloadBaseModel): + mention_type: Literal[BotAPIMentionTypes.USER] + mention_id: UUID + mention_data: BotXAPIPersonalMentionData + + def to_botx_embed_mention_format(self) -> str: + return f"@{{mention:{self.mention_id}}}" + + +class BotXAPIContactMention(UnverifiedPayloadBaseModel): + mention_type: Literal[BotAPIMentionTypes.CONTACT] + mention_id: UUID + mention_data: BotXAPIPersonalMentionData + + def to_botx_embed_mention_format(self) -> str: + return f"@@{{mention:{self.mention_id}}}" + + +class BotXAPIGroupMentionData(UnverifiedPayloadBaseModel): + group_chat_id: UUID + name: Missing[str] + + +class BotXAPIChatMention(UnverifiedPayloadBaseModel): + mention_type: Literal[BotAPIMentionTypes.CHAT] + mention_id: UUID + mention_data: BotXAPIGroupMentionData + + def to_botx_embed_mention_format(self) -> str: + return f"##{{mention:{self.mention_id}}}" + + +class BotXAPIChannelMention(UnverifiedPayloadBaseModel): + mention_type: Literal[BotAPIMentionTypes.CHANNEL] + mention_id: UUID + mention_data: BotXAPIGroupMentionData + + def to_botx_embed_mention_format(self) -> str: + return f"##{{mention:{self.mention_id}}}" + + +class BotXAPIAllMention(UnverifiedPayloadBaseModel): + mention_type: Literal[BotAPIMentionTypes.ALL] + mention_id: UUID + + def to_botx_embed_mention_format(self) -> str: + return f"@{{mention:{self.mention_id}}}" + + +BotXAPIMention = Union[ + BotXAPIUserMention, + BotXAPIContactMention, + BotXAPIChatMention, + BotXAPIChannelMention, + BotXAPIAllMention, +] + + +def build_botx_api_embed_mention( + mention_dict: Dict[str, str], +) -> BotXAPIMention: + mention_type = MentionTypes(mention_dict["mention_type"]) + mentioned_entity_id = mention_dict["mentioned_entity_id"] + # re match will have "" if mention_name not passed + mention_name = mention_dict["mention_name"] or Undefined + + if mention_type == MentionTypes.USER: + return BotXAPIUserMention( + mention_type=convert_mention_type_from_domain(mention_type), + mention_id=uuid4(), + mention_data=BotXAPIPersonalMentionData( + user_huid=UUID(mentioned_entity_id), + name=mention_name, + ), + ) + + if mention_type == MentionTypes.CONTACT: + return BotXAPIContactMention( + mention_type=convert_mention_type_from_domain(mention_type), + mention_id=uuid4(), + mention_data=BotXAPIPersonalMentionData( + user_huid=UUID(mentioned_entity_id), + name=mention_name, + ), + ) + + if mention_type == MentionTypes.CHAT: + return BotXAPIChatMention( + mention_type=convert_mention_type_from_domain(mention_type), + mention_id=uuid4(), + mention_data=BotXAPIGroupMentionData( + group_chat_id=UUID(mentioned_entity_id), + name=mention_name, + ), + ) + + if mention_type == MentionTypes.CHANNEL: + return BotXAPIChannelMention( + mention_type=convert_mention_type_from_domain(mention_type), + mention_id=uuid4(), + mention_data=BotXAPIGroupMentionData( + group_chat_id=UUID(mentioned_entity_id), + name=mention_name, + ), + ) + + if mention_type == MentionTypes.ALL: + return BotXAPIAllMention( + mention_type=convert_mention_type_from_domain(mention_type), + mention_id=uuid4(), + ) + + raise NotImplementedError + + +EMBED_MENTION_RE = re.compile( + ( + "" + "(?P.+?):" + r"(?P[0-9a-f\-]*?):" + "(?P.*?)" + r"<\/embed_mention>" + ), +) + + +def find_and_replace_embed_mentions(body: str) -> Tuple[str, List[BotXAPIMention]]: + mentions = [] + + for match in EMBED_MENTION_RE.finditer(body): + mention_dict = match.groupdict() + embed_mention = match.group(0) + + mention = build_botx_api_embed_mention(mention_dict) + body = body.replace(embed_mention, mention.to_botx_embed_mention_format(), 1) + + mentions.append(mention) + + return body, mentions diff --git a/botx/models/message/message_status.py b/botx/models/message/message_status.py new file mode 100644 index 00000000..def98483 --- /dev/null +++ b/botx/models/message/message_status.py @@ -0,0 +1,12 @@ +from dataclasses import dataclass +from datetime import datetime +from typing import Dict, List +from uuid import UUID + + +@dataclass +class MessageStatus: + group_chat_id: UUID + sent_to: List[UUID] + read_by: Dict[UUID, datetime] + received_by: Dict[UUID, datetime] diff --git a/botx/models/message/outgoing_message.py b/botx/models/message/outgoing_message.py new file mode 100644 index 00000000..d3aa51ec --- /dev/null +++ b/botx/models/message/outgoing_message.py @@ -0,0 +1,24 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Union +from uuid import UUID + +from botx.missing import Missing, Undefined +from botx.models.attachments import IncomingFileAttachment, OutgoingAttachment +from botx.models.message.markup import BubbleMarkup, KeyboardMarkup + + +@dataclass +class OutgoingMessage: + bot_id: UUID + chat_id: UUID + body: str + metadata: Missing[Dict[str, Any]] = Undefined + bubbles: Missing[BubbleMarkup] = Undefined + keyboard: Missing[KeyboardMarkup] = Undefined + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined + silent_response: Missing[bool] = Undefined + markup_auto_adjust: Missing[bool] = Undefined + recipients: Missing[List[UUID]] = Undefined + stealth_mode: Missing[bool] = Undefined + send_push: Missing[bool] = Undefined + ignore_mute: Missing[bool] = Undefined diff --git a/botx/models/message/reply.py b/botx/models/message/reply.py new file mode 100644 index 00000000..a18b3c7f --- /dev/null +++ b/botx/models/message/reply.py @@ -0,0 +1,28 @@ +from dataclasses import dataclass +from typing import List, Literal +from uuid import UUID + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.enums import BotAPIEntityTypes +from botx.models.message.mentions import BotAPIMentionData, MentionList + + +@dataclass +class Reply: + author_id: UUID + sync_id: UUID + body: str + mentions: MentionList + + +class BotAPIReplyData(VerifiedPayloadBaseModel): + source_sync_id: UUID + sender: UUID + body: str + mentions: List[BotAPIMentionData] + # Ignoring attachments cause they don't have content + + +class BotAPIReply(VerifiedPayloadBaseModel): + type: Literal[BotAPIEntityTypes.REPLY] + data: BotAPIReplyData diff --git a/botx/models/message/reply_message.py b/botx/models/message/reply_message.py new file mode 100644 index 00000000..b9034eaa --- /dev/null +++ b/botx/models/message/reply_message.py @@ -0,0 +1,23 @@ +from dataclasses import dataclass +from typing import Any, Dict, Union +from uuid import UUID + +from botx.missing import Missing, Undefined +from botx.models.attachments import IncomingFileAttachment, OutgoingAttachment +from botx.models.message.markup import BubbleMarkup, KeyboardMarkup + + +@dataclass +class ReplyMessage: + bot_id: UUID + sync_id: UUID + body: str + metadata: Missing[Dict[str, Any]] = Undefined + bubbles: Missing[BubbleMarkup] = Undefined + keyboard: Missing[KeyboardMarkup] = Undefined + file: Missing[Union[IncomingFileAttachment, OutgoingAttachment]] = Undefined + silent_response: Missing[bool] = Undefined + markup_auto_adjust: Missing[bool] = Undefined + stealth_mode: Missing[bool] = Undefined + send_push: Missing[bool] = Undefined + ignore_mute: Missing[bool] = Undefined diff --git a/botx/models/messages/__init__.py b/botx/models/messages/__init__.py deleted file mode 100644 index 5a28247a..00000000 --- a/botx/models/messages/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Entities for messages: incoming or outgoing.""" diff --git a/botx/models/messages/incoming_message.py b/botx/models/messages/incoming_message.py deleted file mode 100644 index 3b8dbefb..00000000 --- a/botx/models/messages/incoming_message.py +++ /dev/null @@ -1,204 +0,0 @@ -"""Definition of messages received by bot or sent by it.""" - -from typing import Any, Dict, List, Optional, Tuple, Union -from uuid import UUID - -from pydantic import BaseConfig, BaseModel, Field, validator - -from botx.models import events -from botx.models.attachments import AttachList -from botx.models.entities import EntityList -from botx.models.enums import ChatTypes, ClientPlatformEnum, CommandTypes -from botx.models.files import File, MetaFile - -CommandDataType = Union[ - events.ChatCreatedEvent, - events.AddedToChatEvent, - events.DeletedFromChatEvent, - events.LeftFromChatEvent, - events.InternalBotNotificationEvent, - events.CTSLoginEvent, - events.CTSLogoutEvent, - events.SmartAppEvent, - Dict[str, Any], -] - - -class Command(BaseModel): - """Command that should be proceed by bot.""" - - #: incoming text message. - body: str - - #: was command received from user or this is system event. - command_type: CommandTypes - - #: command payload. - data: CommandDataType = {} # noqa: WPS110 - - #: command metadata. - metadata: Dict[str, Any] = {} - - @property - def command(self) -> str: - """First word of body that was sent to bot.""" - return self.body.split(" ", 1)[0] - - @property - def arguments(self) -> Tuple[str, ...]: - """Words that are passed after command.""" - words = (word for word in self.body.split(" ")[1:]) - arguments = (arg for arg in words if arg and not arg.isspace()) - - return tuple(arguments) - - @property - def single_argument(self) -> str: - """Line that passed after command.""" - body_len = len(self.command) - return self.body[body_len:].strip() - - @property - def data_dict(self) -> dict: - """Command data as dictionary.""" - if isinstance(self.data, dict): - return self.data - return self.data.dict() - - -class DeviceMeta(BaseModel): - """User device metadata.""" - - #: could send pushes. - pushes: Optional[bool] - - #: user timezone. - timezone: Optional[str] - - #: app permissions (microphone, camera, etc.) - permissions: Optional[Dict[str, Any]] - - -class Sender(BaseModel): - """User that sent message to bot.""" - - #: user id. - user_huid: Optional[UUID] - - #: chat id. - group_chat_id: Optional[UUID] - - #: type of chat. - chat_type: Optional[ChatTypes] - - #: AD login of user. - ad_login: Optional[str] - - #: AD domain of user. - ad_domain: Optional[str] - - #: username of user. - username: Optional[str] - - #: is user admin of chat. - is_admin: Optional[bool] - - #: is user creator of chat. - is_creator: Optional[bool] - - #: device brand. - manufacturer: Optional[str] - - #: device name. - device: Optional[str] - - #: device Operating System. - device_software: Optional[str] - - #: device metadata. - device_meta: Optional[DeviceMeta] - - #: client platform name. - platform: Optional[ClientPlatformEnum] - - #: platform package ID with app data and device. - platform_package_id: Optional[str] - - #: Express app version. - app_version: Optional[str] - - #: session locale. - locale: Optional[str] - - #: host from which user sent message. - host: str - - @property - def upn(self) -> Optional[str]: - """User principal name. - - https://docs.microsoft.com/en-us/windows/win32/adschema/a-userprincipalname - """ - if self.ad_login and self.ad_domain: - return "{0}@{1}".format(self.ad_login, self.ad_domain) - - return None - - -class IncomingMessage(BaseModel): - """ - Message that was received by bot and should be handled. - - Warning: - `file` is deprecated field for botx api v4+. - """ - - #: message event id on which bot should answer. - sync_id: UUID - - #: command for bot. - command: Command - - #: file attached to message. - file: Optional[File] = None - - #: meta info for downloading files - async_files: List[MetaFile] = Field(default_factory=list) - - #: information about user from which message was received. - user: Sender = Field(..., alias="from") - - #: ID of message whose ui element was triggered to send this message. - source_sync_id: Optional[UUID] = None - - #: id of bot that should handle message. - bot_id: UUID - - #: additional entities that can be received by bot. - entities: EntityList = Field([]) - - #: attached documents and files to message. - attachments: AttachList = Field([]) - - class Config(BaseConfig): - allow_population_by_field_name = True - - @validator("file", always=True, pre=True) - def skip_file_validation( - cls, - file: Optional[Union[dict, File]], # noqa: N805 - ) -> Optional[File]: - """Skip validation for incoming file since users have not such limits as bot. - - Arguments: - file: file data that should be used for building file instance. - - Returns: - Constructed file. - """ - if isinstance(file, File): - return file - elif file is not None: - return File.construct(**file) - - return None diff --git a/botx/models/messages/message.py b/botx/models/messages/message.py deleted file mode 100644 index 88e98aec..00000000 --- a/botx/models/messages/message.py +++ /dev/null @@ -1,174 +0,0 @@ -"""Definition of message object that is used in all bot handlers.""" -from __future__ import annotations - -from functools import partial -from typing import Any, Dict, List, Optional, Type -from uuid import UUID - -from botx.bots import bots -from botx.models.attachments import AttachList -from botx.models.chats import ChatTypes -from botx.models.datastructures import State -from botx.models.entities import EntityList -from botx.models.enums import CommandTypes -from botx.models.files import File, MetaFile -from botx.models.messages.incoming_message import Command, IncomingMessage, Sender -from botx.models.messages.sending.credentials import SendingCredentials - - -class _ProxyProperty: - def __init__(self, proxy_attribute_name: str, *nested_fields: str) -> None: - self.proxy_attribute_name = proxy_attribute_name - self.nested_fields = list(nested_fields) - - def __get__(self, instance: Message, _owner: Type[Message]) -> Any: - proxy_object = getattr(instance, self.proxy_attribute_name) - accessed_result = proxy_object - for nested_field in self.nested_fields: - accessed_result = getattr(accessed_result, nested_field) - return accessed_result - - def __set_name__(self, _owner: Type[Message], name: str) -> None: - self.nested_fields.append(name) - - -def _proxy_property(proxy_attribute_name: str, *nested_fields: str) -> Any: - return _ProxyProperty(proxy_attribute_name, *nested_fields) - - -_message_proxy_property = partial(_proxy_property, "incoming_message") -_user_proxy_property = partial(_message_proxy_property, "user") - - -class Message: - """Message that is used in handlers.""" - - #: incoming message from BotX. - incoming_message: IncomingMessage - - #: bot that handles this message processing. - bot: "bots.Bot" - - #: state of message during processing. - state: State - - #: ID of message event. - sync_id: UUID = _message_proxy_property() - - #: ID of message whose ui element was triggered to send this message. - source_sync_id: Optional[UUID] = _message_proxy_property() - - #: ID of bot that handles message in Express. - bot_id: UUID = _message_proxy_property() - - #: access to command information. - command: Command = _message_proxy_property() - - #: command body. - body: str = _message_proxy_property("command") - - #: command metadata. - metadata: dict = _message_proxy_property("command") - - #: file from message. - file: Optional[File] = _message_proxy_property() - - #: Meta for download files. - async_files: List[MetaFile] = _message_proxy_property() - - #: attachment from message v4+ - attachments: AttachList = _message_proxy_property() - - #: information about user that sent message. - user: Sender = _message_proxy_property() - - #: HUID of user. - user_huid: Optional[UUID] = _user_proxy_property() - - #: AD login of user. - ad_login: Optional[str] = _user_proxy_property() - - #: AD domain of user. - ad_domain: Optional[str] = _user_proxy_property() - - #: ID of chat from which message was received. - group_chat_id: Optional[UUID] = _user_proxy_property() - - #: type of chat. - chat_type: Optional[ChatTypes] = _user_proxy_property() - - #: host of CTS from which message was received. - host: str = _user_proxy_property() - - #: external entities in message (mentions, forwards, etc) - entities: EntityList = _message_proxy_property() - - #: credentials from message for using in requests. - credentials: SendingCredentials - - #: flag for marking that message was received from button. - sent_from_button: bool - - def __init__(self, message: IncomingMessage, bot: "bots.Bot") -> None: - """Initialize and update fields. - - Arguments: - message: incoming message. - bot: bot that handles message processing. - """ - self.incoming_message = message - self.bot = bot - - self.state = State() - - self.credentials = SendingCredentials( - sync_id=self.sync_id, - bot_id=self.bot_id, - host=self.host, - chat_id=self.group_chat_id, - ) - - @classmethod - def from_dict(cls, message: dict, bot: "bots.Bot") -> Message: - """Parse incoming dict into message. - - Arguments: - message: incoming message to bot as dictionary. - bot: bot that handles message. - - Returns: - Parsed message. - """ - return cls(IncomingMessage(**message), bot) - - @property - def is_forward(self) -> bool: - """Check this message on forwarding. - - Returns: - bool: True if message is forward else False - """ - try: # noqa: WPS503 - self.entities.forward # noqa: WPS428 - except AttributeError: - return False - else: - return True - - @property - def is_system_event(self) -> bool: - """Check if message is a system event. - - Returns: - bool: Result of check. - """ - return self.command.command_type == CommandTypes.system - - @property - def data(self) -> Dict[Any, Any]: # noqa: WPS110 - """Concatenated metadata and data from UI element. - - Returns: - dict: Concatenated metadata and data. - """ - return {**self.metadata, **self.incoming_message.command.data_dict} diff --git a/botx/models/messages/sending/__init__.py b/botx/models/messages/sending/__init__.py deleted file mode 100644 index c62ba5ce..00000000 --- a/botx/models/messages/sending/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition for entities that are used in sending messages.""" diff --git a/botx/models/messages/sending/credentials.py b/botx/models/messages/sending/credentials.py deleted file mode 100644 index 86540b58..00000000 --- a/botx/models/messages/sending/credentials.py +++ /dev/null @@ -1,28 +0,0 @@ -"""Definition for sending credentials.""" - -from typing import Optional -from uuid import UUID - -from botx.models.base import BotXBaseModel - - -class SendingCredentials(BotXBaseModel): - """Credentials that are required to send command or notification result.""" - - #: message event id. - sync_id: Optional[UUID] = None - - #: id of message that will be sent. - message_id: Optional[UUID] = None - - #: chat id in which bot should send message. - chat_id: Optional[UUID] = None - - #: bot that handles message. - bot_id: Optional[UUID] = None - - #: host on which bot answers. - host: Optional[str] = None - - #: token that is used for bot authorization on requests to BotX API. - token: Optional[str] = None diff --git a/botx/models/messages/sending/markup.py b/botx/models/messages/sending/markup.py deleted file mode 100644 index 8b5ba892..00000000 --- a/botx/models/messages/sending/markup.py +++ /dev/null @@ -1,155 +0,0 @@ -"""Definition for markup attached to sent message.""" - -from typing import List, Optional, Type, TypeVar - -from botx.models.base import BotXBaseModel -from botx.models.buttons import BubbleElement, Button, ButtonOptions, KeyboardElement - -TUIElement = TypeVar("TUIElement", bound=Button) - - -class MessageMarkup(BotXBaseModel): - """Collection for bubbles and keyboard with some helper methods.""" - - #: bubbles that will be attached to message. - bubbles: List[List[BubbleElement]] = [] - - #: keyboard elements that will be attached to message. - keyboard: List[List[KeyboardElement]] = [] - - def add_bubble( # noqa: WPS211 - self, - command: str, - label: Optional[str] = None, - data: Optional[dict] = None, # noqa: WPS110 - options: Optional[ButtonOptions] = None, - *, - new_row: bool = True, - ) -> None: - """Add new bubble button to markup. - - Arguments: - command: command that will be triggered on bubble click. - label: label that will be shown on bubble. - data: payload that will be attached to bubble. - options: add special effects to bubble. - new_row: place bubble on new row or on current. - """ - self._add_ui_element( - ui_cls=BubbleElement, - ui_array=self.bubbles, - command=command, - label=label, - data=data, - opts=options, - new_row=new_row, - ) - - def add_bubble_element( - self, - element: BubbleElement, - *, - new_row: bool = True, - ) -> None: - """Add new button to markup from existing element. - - Arguments: - element: existed bubble element. - new_row: place bubble on new row or on current. - """ - self._add_ui_element( - ui_cls=BubbleElement, - ui_array=self.bubbles, - command=element.command, - label=element.label, - data=element.data, - opts=element.opts, - new_row=new_row, - ) - - def add_keyboard_button( # noqa: WPS211 - self, - command: str, - label: Optional[str] = None, - data: Optional[dict] = None, # noqa: WPS110 - options: Optional[ButtonOptions] = None, - *, - new_row: bool = True, - ) -> None: - """Add new keyboard button to markup. - - Arguments: - command: command that will be triggered on keyboard click. - label: label that will be shown on keyboard button. - data: payload that will be attached to keyboard. - options: add special effects to keyboard button. - new_row: place keyboard on new row or on current. - """ - self._add_ui_element( - ui_cls=KeyboardElement, - ui_array=self.keyboard, - command=command, - label=label, - data=data, - opts=options, - new_row=new_row, - ) - - def add_keyboard_button_element( - self, - element: KeyboardElement, - *, - new_row: bool = True, - ) -> None: - """Add new keyboard button to markup from existing element. - - Arguments: - element: existed keyboard button element. - new_row: place keyboard button on new row or on current. - """ - self._add_ui_element( - ui_cls=KeyboardElement, - ui_array=self.keyboard, - command=element.command, - label=element.label, - data=element.data, - opts=element.opts, - new_row=new_row, - ) - - def _add_ui_element( # noqa: WPS211 - self, - ui_cls: Type[TUIElement], - ui_array: List[List[TUIElement]], - command: str, - label: Optional[str] = None, - data: Optional[dict] = None, # noqa: WPS110 - opts: Optional[ButtonOptions] = None, - new_row: bool = True, - ) -> None: - """Add new button to bubble or keyboard arrays. - - Arguments: - ui_cls: UIElement instance that should be added to array. - ui_array: storage for ui elements. - command: command that will be triggered on ui element click. - label: label that will be shown on ui element. - data: payload that will be attached to ui element. - opts: add special effects ui element. - new_row: place ui element on new row or on current. - """ - element = ui_cls( - command=command, - label=label, - data=(data or {}), - opts=(opts or ButtonOptions()), - ) - - if new_row: - ui_array.append([element]) - return - - if not ui_array: - ui_array.append([]) - - ui_array[-1].append(element) diff --git a/botx/models/messages/sending/message.py b/botx/models/messages/sending/message.py deleted file mode 100644 index 6b8aa470..00000000 --- a/botx/models/messages/sending/message.py +++ /dev/null @@ -1,651 +0,0 @@ -"""Message that is sent from bot.""" -import re -from typing import Any # noqa: WPS235 -from typing import BinaryIO, Dict, List, Optional, TextIO, Tuple, Union, cast -from uuid import UUID - -from botx.models.buttons import ButtonOptions -from botx.models.entities import ChatMention, Mention, UserMention -from botx.models.enums import MentionTypes -from botx.models.files import File -from botx.models.messages.message import Message -from botx.models.messages.sending.credentials import SendingCredentials -from botx.models.messages.sending.markup import MessageMarkup -from botx.models.messages.sending.options import MessageOptions, NotificationOptions -from botx.models.messages.sending.payload import MessagePayload -from botx.models.typing import AvailableRecipients, BubbleMarkup, KeyboardMarkup - -try: - from typing import Final # noqa: WPS433 -except ImportError: - from typing_extensions import Final # type: ignore # noqa: WPS433, WPS440, F401 - -ARGUMENTS_DUPLICATION_ERROR = ( - "{0} can not be passed along with manual validated_values for it" -) - -EMBED_MENTION_TEMPLATE = ( - "" # noqa: WPS326 -) -EMBED_MENTION_RE: Final = re.compile( - r".+?):(?P[0-9a-f\-]+?):" - r"(?P[0-9a-f\-]+?):(?P.+?)??>", # noqa: WPS326 C812 -) - - -# currently I have no idea how to clean this. -class SendingMessage: # noqa: WPS214 - """Message that will be sent by bot.""" - - def __init__( # noqa: WPS211 - self, - *, - text: str = "", - bot_id: Optional[UUID] = None, - host: Optional[str] = None, - sync_id: Optional[UUID] = None, - chat_id: Optional[UUID] = None, - message_id: Optional[UUID] = None, - recipients: Optional[AvailableRecipients] = None, - mentions: Optional[List[Mention]] = None, - bubbles: Optional[BubbleMarkup] = None, - keyboard: Optional[KeyboardMarkup] = None, - notification_options: Optional[NotificationOptions] = None, - file: Optional[File] = None, - credentials: Optional[SendingCredentials] = None, - options: Optional[MessageOptions] = None, - markup: Optional[MessageMarkup] = None, - metadata: Optional[Dict[str, Any]] = None, - embed_mentions: bool = False, - ) -> None: - """Init message with required attributes. - - !!! info - You should pass at least already built credentials or bot_id, host and - one of sync_id, chat_id or chat_ids for message. - !!! info - You can not pass markup along with bubbles or keyboards. You can merge them - manual before or after building message. - !!! info - You can not pass options along with any of recipients, mentions or - notification_options. You can merge them manual before or after building - message. - - Arguments: - text: text for message. - file: file that will be attached to message. - bot_id: bot id. - host: host for message. - sync_id: message event id. - chat_id: chat id. - message_id: custom id of new message. - credentials: message credentials. - bubbles: bubbles that will be attached to message. - keyboard: keyboard elements that will be attached to message. - markup: message markup. - recipients: recipients for message. - mentions: mentions that will be attached to message. - notification_options: configuration for notifications for message. - options: message options. - metadata: message metadata. - embed_mentions: get mentions from text. - """ - self.credentials: SendingCredentials = _build_credentials( - bot_id=bot_id, - host=host, - sync_id=sync_id, - message_id=message_id, - chat_id=chat_id, - credentials=(credentials.copy() if credentials else credentials), - ) - - options = _build_options( - recipients=recipients, - mentions=mentions, - notification_options=notification_options, - options=options, - ) - if embed_mentions: - updated_text, found_mentions = self._find_and_replace_embed_mentions(text) - - text = updated_text - options.mentions = found_mentions - options.raw_mentions = True - - self.payload: MessagePayload = MessagePayload( - text=text, - metadata=metadata or {}, - file=file, - markup=_build_markup(bubbles=bubbles, keyboard=keyboard, markup=markup), - options=options, - ) - - @classmethod - def from_message( - cls, - *, - text: str = "", - file: Optional[File] = None, - message: Message, - embed_mentions: bool = False, - ) -> "SendingMessage": - """Build message for sending from incoming message. - - Arguments: - text: text for message. - file: file attached to message. - message: incoming message. - embed_mentions: get mentions from text. - - Returns: - Built message. - """ - return cls( - text=text, - file=file, - sync_id=message.sync_id, - chat_id=message.group_chat_id, - bot_id=message.bot_id, - host=message.host, - embed_mentions=embed_mentions, - ) - - @classmethod - def make_mention_embeddable(cls, mention: Mention) -> str: - """Get mention as string, which can be embed in text. - - Arguments: - mention: mention for embedding. - - Raises: - NotImplementedError: If unsupported mention type was passed. - - Returns: - Formatted mention. - """ - if mention.mention_type in {MentionTypes.user, MentionTypes.contact}: - assert isinstance(mention.mention_data, UserMention) # for mypy - mentioned_entity_id = mention.mention_data.user_huid - elif mention.mention_type in {MentionTypes.chat, MentionTypes.channel}: - assert isinstance(mention.mention_data, ChatMention) # for mypy - mentioned_entity_id = mention.mention_data.group_chat_id - else: - raise NotImplementedError("Unsupported mention type") - - mention_name = mention.mention_data.name or "" - - return EMBED_MENTION_TEMPLATE.format( - mention_type=mention.mention_type, - mentioned_entity_id=mentioned_entity_id, - mention_id=mention.mention_id, - mention_name=mention_name, - ) - - @classmethod - def build_embeddable_user_mention( - cls, - user_huid: UUID, - name: Optional[str] = None, - mention_id: Optional[UUID] = None, - ) -> str: - """Get user mention as string, which can be embed in text. - - Arguments: - user_huid: user id to mention. - name: for overriding mention name. - mention_id: mention id (if not passed, will be generated). - - Returns: - Formatted mention. - """ - mention = Mention.build_from_values( - MentionTypes.user, - user_huid, - name, - mention_id, - ) - - return cls.make_mention_embeddable(mention) - - @classmethod - def build_embeddable_contact_mention( - cls, - user_huid: UUID, - name: Optional[str] = None, - mention_id: Optional[UUID] = None, - ) -> str: - """Get contact mention as string, which can be embed in text. - - Arguments: - user_huid: user id to mention. - name: for overriding mention name. - mention_id: mention id (if not passed, will be generated). - - Returns: - Formatted mention. - """ - mention = Mention.build_from_values( - MentionTypes.contact, - user_huid, - name, - mention_id, - ) - - return cls.make_mention_embeddable(mention) - - @classmethod - def build_embeddable_chat_mention( - cls, - group_chat_id: UUID, - name: Optional[str] = None, - mention_id: Optional[UUID] = None, - ) -> str: - """Get chat mention as string, which can be embed in text. - - Arguments: - group_chat_id: chat id to mention. - name: for overriding mention name. - mention_id: mention id (if not passed, will be generated). - - Returns: - Formatted mention. - """ - mention = Mention.build_from_values( - MentionTypes.chat, - group_chat_id, - name, - mention_id, - ) - - return cls.make_mention_embeddable(mention) - - @classmethod - def build_embeddable_channel_mention( - cls, - group_chat_id: UUID, - name: Optional[str] = None, - mention_id: Optional[UUID] = None, - ) -> str: - """Get channel mention as string, which can be embed in text. - - Arguments: - group_chat_id: channel id to mention. - name: for overriding mention name. - mention_id: mention id (if not passed, will be generated). - - Returns: - Formatted mention. - """ - mention = Mention.build_from_values( - MentionTypes.channel, - group_chat_id, - name, - mention_id, - ) - - return cls.make_mention_embeddable(mention) - - @property - def text(self) -> str: - """Text in message.""" - return self.payload.text - - @text.setter - def text(self, text: str) -> None: - """Text in message.""" - self.payload.text = text - - @property - def metadata(self) -> Dict[str, Any]: - """Metadata in message.""" - return self.payload.metadata - - @metadata.setter - def metadata(self, metadata: Dict[str, Any]) -> None: - self.payload.metadata = metadata - - @property - def file(self) -> Optional[File]: - """File attached to message.""" - return self.payload.file - - @file.setter - def file(self, file: File) -> None: - """File attached to message.""" - self.payload.file = file - - @property - def markup(self) -> MessageMarkup: - """Message markup.""" - return self.payload.markup - - @markup.setter - def markup(self, markup: MessageMarkup) -> None: - """Message markup.""" - self.payload.markup = markup - - @property - def options(self) -> MessageOptions: - """Message options.""" - return self.payload.options - - @options.setter - def options(self, options: MessageOptions) -> None: - """Message options.""" - self.payload.options = options - - @property - def sync_id(self) -> Optional[UUID]: - """Event id on which message should answer.""" - return self.credentials.sync_id - - @sync_id.setter - def sync_id(self, sync_id: UUID) -> None: - """Event id on which message should answer.""" - self.credentials.sync_id = sync_id - - @property - def chat_id(self) -> Optional[UUID]: - """Chat id in which message should be sent.""" - return self.credentials.chat_id - - @chat_id.setter - def chat_id(self, chat_id: UUID) -> None: - """Chat id in which message should be sent.""" - self.credentials.chat_id = chat_id - - @property - def bot_id(self) -> UUID: - """Bot id that handles message.""" - return cast(UUID, self.credentials.bot_id) - - @bot_id.setter - def bot_id(self, bot_id: UUID) -> None: - """Bot id that handles message.""" - self.credentials.bot_id = bot_id - - @property - def host(self) -> str: - """Host where BotX API places.""" - return cast(str, self.credentials.host) - - @host.setter - def host(self, host: str) -> None: - """Host where BotX API places.""" - self.credentials.host = host - - def add_file( - self, - file: Union[TextIO, BinaryIO, File], - filename: Optional[str] = None, - ) -> None: - """Attach file to message. - - Arguments: - file: file that should be attached to the message. - filename: name for file that will be used if if can not be retrieved from - file. - """ - if isinstance(file, File): - file.file_name = filename or file.file_name - self.payload.file = file - else: - self.payload.file = File.from_file(file, filename=filename) - - def mention_user(self, user_huid: UUID, name: Optional[str] = None) -> None: - """Mention user in message. - - Arguments: - user_huid: id of user that should be mentioned. - name: name that will be shown. - """ - self.payload.options.mentions.append( - Mention(mention_data=UserMention(user_huid=user_huid, name=name)), - ) - - def mention_contact(self, user_huid: UUID, name: Optional[str] = None) -> None: - """Mention contact in message. - - Arguments: - user_huid: id of user that should be mentioned. - name: name that will be shown. - """ - self.payload.options.mentions.append( - Mention( - mention_data=UserMention(user_huid=user_huid, name=name), - mention_type=MentionTypes.contact, - ), - ) - - def mention_chat(self, group_chat_id: UUID, name: Optional[str] = None) -> None: - """Mention chat in message. - - Arguments: - group_chat_id: id of chat that should be mentioned. - name: name that will be shown. - """ - self.payload.options.mentions.append( - Mention( - mention_data=ChatMention(group_chat_id=group_chat_id, name=name), - mention_type=MentionTypes.chat, - ), - ) - - def add_recipient(self, recipient: UUID) -> None: - """Add new user that will receive message. - - Arguments: - recipient: recipient for message. - """ - if self.payload.options.recipients == "all": - self.payload.options.recipients = [] - - self.payload.options.recipients.append(recipient) - - def add_recipients(self, recipients: List[UUID]) -> None: - """Add list of recipients that should receive message. - - Arguments: - recipients: recipients for message. - """ - if self.payload.options.recipients == "all": - self.payload.options.recipients = [] - - self.payload.options.recipients.extend(recipients) - - def add_bubble( # noqa: WPS211 - self, - command: str, - label: Optional[str] = None, - data: Optional[dict] = None, # noqa: WPS110 - options: Optional[ButtonOptions] = None, - *, - new_row: bool = True, - ) -> None: - """Add new bubble button to message markup. - - Arguments: - command: command that will be triggered on bubble click. - label: label that will be shown on bubble. - data: payload that will be attached to bubble. - options: add special effects to bubble. - new_row: place bubble on new row or on current. - """ - self.payload.markup.add_bubble(command, label, data, options, new_row=new_row) - - def add_keyboard_button( # noqa: WPS211 - self, - command: str, - label: Optional[str] = None, - data: Optional[dict] = None, # noqa: WPS110 - options: Optional[ButtonOptions] = None, - *, - new_row: bool = True, - ) -> None: - """Add new keyboard button to message markup. - - Arguments: - command: command that will be triggered on keyboard click. - label: label that will be shown on keyboard button. - data: payload that will be attached to keyboard. - options: add special effects to keyboard button. - new_row: place keyboard on new row or on current. - """ - self.payload.markup.add_keyboard_button( - command, - label, - data, - options, - new_row=new_row, - ) - - def show_notification(self, show: bool) -> None: - """Show notification about message. - - Arguments: - show: show notification about message. - """ - self.payload.options.notifications.send = show - - def force_notification(self, force: bool) -> None: - """Break mute on bot messages. - - Arguments: - force: break mute on bot messages. - """ - self.payload.options.notifications.force_dnd = force - - def _find_and_replace_embed_mentions( # noqa: WPS210 - self, - text: str, - ) -> Tuple[str, List[Mention]]: - mentions = [] - - match = EMBED_MENTION_RE.search(text) - while match: - mention_dict = match.groupdict() - embed_mention = match.group(0) - - mention = Mention.build_from_values( - MentionTypes(mention_dict["mention_type"]), - UUID(mention_dict["mentioned_entity_id"]), - mention_dict["name"], - UUID(mention_dict["mention_id"]), - ) - - text = text.replace(embed_mention, mention.to_botx_format()) - mentions.append(mention) - - match = EMBED_MENTION_RE.search(text) - - return text, mentions - - -def _build_credentials( # noqa: WPS211 - bot_id: Optional[UUID] = None, - host: Optional[str] = None, - sync_id: Optional[UUID] = None, - message_id: Optional[UUID] = None, - chat_id: Optional[UUID] = None, - credentials: Optional[SendingCredentials] = None, -) -> SendingCredentials: - """Build credentials for message. - - Arguments: - bot_id: bot id. - host: host for message. - sync_id: message event id. - message_id: id of new message. - chat_id: chat id. - credentials: message credentials. - - Returns: - Credentials for message. - - Raises: - AssertionError: raised if credentials were passed with separate parameters. - """ - if bot_id and host: - if credentials is not None: - raise AssertionError( - ARGUMENTS_DUPLICATION_ERROR.format("MessageCredentials"), - ) - - return SendingCredentials( - bot_id=bot_id, - host=host, - sync_id=sync_id, - chat_id=chat_id, - message_id=message_id, - ) - - if credentials is None: - raise AssertionError( - "MessageCredentials or manual validated_values should be passed", - ) - - if credentials.message_id is None: - credentials.message_id = message_id - - return credentials - - -def _build_markup( - bubbles: Optional[BubbleMarkup] = None, - keyboard: Optional[KeyboardMarkup] = None, - markup: Optional[MessageMarkup] = None, -) -> MessageMarkup: - """Build markup for message. - - Arguments: - bubbles: bubbles that will be attached to message. - keyboard: keyboard elements that will be attached to message. - markup: message markup. - - Returns: - Markup for message. - - Raises: - AssertionError: raised if markup were passed with separate parameters. - """ - if bubbles is not None or keyboard is not None: - if markup is not None: - raise AssertionError( - "Markup can not be passed along with bubbles or keyboard elements", - ) - return MessageMarkup(bubbles=bubbles or [], keyboard=keyboard or []) - - return markup or MessageMarkup() - - -def _build_options( - recipients: Optional[AvailableRecipients] = None, - mentions: Optional[List[Mention]] = None, - notification_options: Optional[NotificationOptions] = None, - options: Optional[MessageOptions] = None, -) -> MessageOptions: - """Build options for message. - - Arguments: - recipients: recipients for message. - mentions: mentions that will be attached to message. - notification_options: configuration for notifications for message. - options: message options. - - Returns: - Options for message. - - Raises: - AssertionError: raised if options were passed with separate parameters. - """ - if mentions or recipients or notification_options: - if options is not None: - raise AssertionError(ARGUMENTS_DUPLICATION_ERROR.format("MessageOptions")) - return MessageOptions( - recipients=recipients or "all", - mentions=mentions or [], - notifications=notification_options or NotificationOptions(), - ) - - return options or MessageOptions() diff --git a/botx/models/messages/sending/options.py b/botx/models/messages/sending/options.py deleted file mode 100644 index 962d7e62..00000000 --- a/botx/models/messages/sending/options.py +++ /dev/null @@ -1,46 +0,0 @@ -"""Special options for message.""" - -from typing import List - -from botx.models.base import BotXBaseModel -from botx.models.entities import Mention -from botx.models.typing import AvailableRecipients - - -class ResultPayloadOptions(BotXBaseModel): - """Options for `notification` and `command_result` API entities.""" - - #: don't show next user's input in chat - silent_response: bool = False - - -class NotificationOptions(BotXBaseModel): - """Configurations for message notifications.""" - - #: show notification about message. - send: bool = True - - #: break mute on bot messages. - force_dnd: bool = False - - -class MessageOptions(BotXBaseModel): - """Message options configuration.""" - - #: users that should receive message. - recipients: AvailableRecipients = "all" - - #: attached to message mentions. - mentions: List[Mention] = [] - - #: don't show next user's input in chat - silent_response: bool = False - - #: deliver message only if stealth mode enabled - stealth_mode: bool = False - - #: use in-text mentions - raw_mentions: bool = False - - #: notification configuration. - notifications: NotificationOptions = NotificationOptions() diff --git a/botx/models/messages/sending/payload.py b/botx/models/messages/sending/payload.py deleted file mode 100644 index b61b30c4..00000000 --- a/botx/models/messages/sending/payload.py +++ /dev/null @@ -1,90 +0,0 @@ -"""Payload for messages.""" -from __future__ import annotations - -from typing import Any, Dict, List, Optional - -from pydantic import Field - -from botx.models.base import BotXBaseModel -from botx.models.constants import MAXIMUM_TEXT_LENGTH -from botx.models.entities import Mention -from botx.models.files import File -from botx.models.messages.sending.markup import MessageMarkup -from botx.models.messages.sending.options import MessageOptions, NotificationOptions -from botx.models.typing import BubbleMarkup, KeyboardMarkup - - -class MessagePayload(BotXBaseModel): - """Message payload configuration.""" - - #: message text. - text: str = Field("", max_length=MAXIMUM_TEXT_LENGTH) - - #: message metadata. - metadata: Dict[str, Any] = {} - - #: file attached to message. - file: Optional[File] = None - - #: message markup. - markup: MessageMarkup = MessageMarkup() - - #: message configuration. - options: MessageOptions = MessageOptions() - - -class UpdatePayload(BotXBaseModel): - """Payload for message edition.""" - - #: new message text. - text: Optional[str] = Field(None, max_length=MAXIMUM_TEXT_LENGTH) - - #: file attached to message. - file: Optional[File] = None - - #: new message bubbles. - keyboard: Optional[KeyboardMarkup] = None - - #: new message keyboard. - bubbles: Optional[BubbleMarkup] = None - - #: new message mentions. - mentions: Optional[List[Mention]] = None - - #: new message options. - opts: Optional[NotificationOptions] = None - - #: message metadata. - metadata: Optional[Dict[str, Any]] = None - - @property - def markup(self) -> MessageMarkup: - """Markup for edited message.""" - return MessageMarkup(bubbles=self.bubbles or [], keyboard=self.keyboard or []) - - def set_markup(self, markup: MessageMarkup) -> None: - """Markup for edited message. - - Arguments: - markup: markup that should be applied to payload. - """ - self.bubbles = markup.bubbles - self.keyboard = markup.keyboard - - @classmethod - def from_sending_payload(cls, payload: MessagePayload) -> UpdatePayload: - """Create new update payload from existing payload for new message. - - Arguments: - payload: payload that can be used for sending new message. - - Returns: - Created payload for update. - """ - update = cls() - update.text = payload.text or None - update.set_markup(payload.markup) - update.mentions = payload.options.mentions - update.file = payload.file - update.metadata = payload.metadata - return update diff --git a/botx/models/method_callbacks.py b/botx/models/method_callbacks.py new file mode 100644 index 00000000..dc248da9 --- /dev/null +++ b/botx/models/method_callbacks.py @@ -0,0 +1,21 @@ +from typing import Any, Dict, List, Literal, Union +from uuid import UUID + +from botx.models.api_base import VerifiedPayloadBaseModel + + +class BotAPIMethodSuccessfulCallback(VerifiedPayloadBaseModel): + sync_id: UUID + status: Literal["ok"] + result: Dict[str, Any] + + +class BotAPIMethodFailedCallback(VerifiedPayloadBaseModel): + sync_id: UUID + status: Literal["error"] + reason: str + errors: List[str] + error_data: Dict[str, Any] + + +BotXMethodCallback = Union[BotAPIMethodSuccessfulCallback, BotAPIMethodFailedCallback] diff --git a/botx/models/smartapps.py b/botx/models/smartapps.py deleted file mode 100644 index 45169b4e..00000000 --- a/botx/models/smartapps.py +++ /dev/null @@ -1,116 +0,0 @@ -"""Definition of smartapp object.""" - -from typing import Any, BinaryIO, Dict, List, Optional, TextIO, Union -from uuid import UUID - -from botx.models.base import BotXBaseModel -from botx.models.files import File, MetaFile -from botx.models.messages.message import Message - - -class SendingSmartAppEvent(BotXBaseModel): - """SmartApp event with data.""" - - #: unique request id - ref: Optional[UUID] = None - - #: smartapp id - smartapp_id: UUID - - #: event data - data: Dict[str, Any] # noqa: WPS110 - - #: event options - opts: Dict[str, Any] = {} - - #: version of protocol smartapp <-> bot - smartapp_api_version: int - - #: smartapp chat - group_chat_id: Optional[UUID] - - #: files - files: List[File] = [] - - #: file's meta to upload - async_files: List[MetaFile] = [] - - @classmethod - def from_message( - cls, - data: Dict[str, Any], # noqa: WPS110 - message: Message, - ) -> "SendingSmartAppEvent": - """Build smartapp event from message. - - Arguments: - data: smartapp's data. - message: incoming message. - - Returns: - Built smartapp event. - """ - return cls( - ref=message.data["ref"], - smartapp_id=message.data["smartapp_id"], - data=data, - opts=message.data["opts"], - smartapp_api_version=message.data["smartapp_api_version"], - group_chat_id=message.group_chat_id, - ) - - def add_file( - self, - file: Union[TextIO, BinaryIO, File], - filename: Optional[str] = None, - ) -> None: - """Attach file to smartapp. - - Arguments: - file: file that should be attached to the message. - filename: name for file that will be used if if can not be retrieved from - file. - """ - if isinstance(file, File): - file.file_name = filename or file.file_name - self.files.append(file) - else: - self.files.append(File.from_file(file, filename=filename)) - - -class SendingSmartAppNotification(BotXBaseModel): - """SmartApp notification with counter.""" - - #: smartapp chat - group_chat_id: Optional[UUID] - - #: unread notifications count - smartapp_counter: int - - #: event options - opts: Dict[str, Any] = {} - - #: version of protocol smartapp <-> bot - smartapp_api_version: int - - @classmethod - def from_message( - cls, - smartapp_counter: int, - message: Message, - ) -> "SendingSmartAppNotification": - """Build smartapp notification from message. - - Arguments: - smartapp_counter: smartapp notification counter. - message: incoming message. - - Returns: - Built smartapp notification. - """ - return cls( - smartapp_counter=smartapp_counter, - opts=message.data["opts"], - smartapp_api_version=message.data["smartapp_api_version"], - group_chat_id=message.group_chat_id, - ) diff --git a/botx/models/status.py b/botx/models/status.py index 7433a49e..f7f0b930 100644 --- a/botx/models/status.py +++ b/botx/models/status.py @@ -1,28 +1,100 @@ -"""Module of model for status recipient.""" -from typing import Optional +from dataclasses import asdict, dataclass +from typing import Any, Dict, List, Literal, NewType, Optional, Union from uuid import UUID -from botx import ChatTypes -from botx.models.base import BotXBaseModel +from pydantic import validator +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.enums import APIChatTypes, ChatTypes, convert_chat_type_to_domain +from botx.models.message.incoming_message import IncomingMessage -class StatusRecipient(BotXBaseModel): - """Model of recipients in status request.""" +BotMenu = NewType("BotMenu", Dict[str, str]) - #: bot that request status + +@dataclass +class StatusRecipient: bot_id: UUID + huid: UUID + ad_login: Optional[str] + ad_domain: Optional[str] + is_admin: Optional[bool] + chat_type: ChatTypes - #: user that request status - user_huid: UUID + @classmethod + def from_incoming_message( + cls, + incoming_message: IncomingMessage, + ) -> "StatusRecipient": + return StatusRecipient( + bot_id=incoming_message.bot.id, + huid=incoming_message.sender.huid, + ad_login=incoming_message.sender.ad_login, + ad_domain=incoming_message.sender.ad_domain, + is_admin=incoming_message.sender.is_chat_admin, + chat_type=incoming_message.chat.type, + ) - #: user's ad_login - ad_login: Optional[str] - #: user's ad_domain +class BotAPIStatusRecipient(VerifiedPayloadBaseModel): + bot_id: UUID + user_huid: UUID + ad_login: Optional[str] ad_domain: Optional[str] - - #: user has admin role is_admin: Optional[bool] + chat_type: APIChatTypes - #: chat type - chat_type: ChatTypes + @validator("ad_login", "ad_domain", "is_admin", pre=True) + @classmethod + def replace_empty_string( + cls, + field_value: Union[str, bool], + ) -> Union[str, bool, None]: + if field_value == "": + return None + + return field_value + + def to_domain(self) -> StatusRecipient: + return StatusRecipient( + bot_id=self.bot_id, + huid=self.user_huid, + ad_login=self.ad_login, + ad_domain=self.ad_domain, + is_admin=self.is_admin, + chat_type=convert_chat_type_to_domain(self.chat_type), + ) + + +@dataclass +class BotAPIBotMenuItem: + description: str + body: str + name: str + + +BotAPIBotMenu = List[BotAPIBotMenuItem] + + +@dataclass +class BotAPIStatusResult: + commands: BotAPIBotMenu + enabled: Literal[True] = True + status_message: Optional[str] = None + + +@dataclass +class BotAPIStatus: + result: BotAPIStatusResult + status: Literal["ok"] = "ok" + + +def build_bot_status_response(bot_menu: BotMenu) -> Dict[str, Any]: + commands = [ + BotAPIBotMenuItem(body=command, name=command, description=description) + for command, description in bot_menu.items() + ] + + status = BotAPIStatus( + result=BotAPIStatusResult(status_message="Bot is working", commands=commands), + ) + return asdict(status) diff --git a/botx/models/stickers.py b/botx/models/stickers.py index c35f1686..05610b6a 100644 --- a/botx/models/stickers.py +++ b/botx/models/stickers.py @@ -1,71 +1,87 @@ -"""Models for stickers.""" -from datetime import datetime +from dataclasses import dataclass from typing import List, Optional from uuid import UUID -from botx.models.base import BotXBaseModel +from botx.async_buffer import AsyncBufferWritable +from botx.bot.contextvars import bot_var -class Pagination(BotXBaseModel): - """Model of pagination.""" - - #: cursor hash - after: Optional[str] +@dataclass +class Sticker: + """Sticker from sticker pack. + Attributes: + id: Sticker id. + emoji: Sticker emoji. + link: Sticker image link. -class Sticker(BotXBaseModel): - """Model of sticker from request by id.""" + """ id: UUID emoji: str - link: str - inserted_at: datetime - updated_at: datetime - deleted_at: Optional[datetime] + image_link: str + async def download( + self, + async_buffer: AsyncBufferWritable, + ) -> None: + bot = bot_var.get() -class StickerFromPack(BotXBaseModel): - """Model of sticker from sticker pack.""" + response = await bot._httpx_client.get(self.image_link) # noqa: WPS437 + response.raise_for_status() - id: UUID - emoji: str - link: str - preview: str + await async_buffer.write(response.content) + await async_buffer.seek(0) -class StickerPackPreview(BotXBaseModel): - """Model of sticker pack from pack list.""" - id: UUID - name: str - preview: Optional[str] - public: Optional[bool] - stickers_count: int - stickers_order: Optional[List[UUID]] - inserted_at: datetime - updated_at: Optional[datetime] - deleted_at: Optional[datetime] +@dataclass +class StickerPack: + """Sticker pack. + Attributes: + id: Sticker pack id. + name: Sticker pack name. + is_public: Is public pack. + stickers: Stickers data. -class StickerPackList(BotXBaseModel): - """Full model of sticker pack list response.""" + """ - #: list of sticker packs - packs: List[StickerPackPreview] + id: UUID + name: str + is_public: bool + stickers: List[Sticker] - #: cursor - pagination: Pagination +@dataclass +class StickerPackFromList: + """Sticker pack from list. -class StickerPack(BotXBaseModel): - """Model of sticker pack from request by id.""" + Attributes: + id: Sticker pack id. + name: Sticker pack name. + is_public: Is public pack + stickers_count: Stickers count in pack + sticker_ids: Stickers ids in pack + + """ id: UUID name: str - public: bool - preview: Optional[str] - stickers_order: Optional[List[UUID]] - stickers: List[Sticker] - inserted_at: datetime - updated_at: datetime - deleted_at: Optional[datetime] + is_public: bool + stickers_count: int + sticker_ids: Optional[List[UUID]] # Can be omitted in result + + +@dataclass +class StickerPackPage: + """Sticker pack page. + + Attributes: + sticker_packs: Sticker pack list. + after: Base64 string for pagination. + + """ + + sticker_packs: List[StickerPackFromList] + after: Optional[str] diff --git a/tests/test_bots/test_bots/__init__.py b/botx/models/system_events/__init__.py similarity index 100% rename from tests/test_bots/test_bots/__init__.py rename to botx/models/system_events/__init__.py diff --git a/botx/models/system_events/added_to_chat.py b/botx/models/system_events/added_to_chat.py new file mode 100644 index 00000000..08654cbc --- /dev/null +++ b/botx/models/system_events/added_to_chat.py @@ -0,0 +1,56 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import BotAPICommandTypes, convert_chat_type_to_domain + + +@dataclass +class AddedToChatEvent(BotCommandBase): + """Event `system:added_to_chat`. + + Attributes: + huids: List of added to chat user huids. + """ + + huids: List[UUID] + chat: Chat + + +class BotAPIAddedToChatData(VerifiedPayloadBaseModel): + added_members: List[UUID] + + +class BotAPIAddedToChatPayload(VerifiedPayloadBaseModel): + body: Literal["system:added_to_chat"] = "system:added_to_chat" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPIAddedToChatData + + +class BotAPIAddedToChat(BotAPIBaseCommand): + payload: BotAPIAddedToChatPayload = Field(..., alias="command") + sender: BotAPIChatContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> AddedToChatEvent: + return AddedToChatEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + huids=self.payload.data.added_members, + chat=Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ), + ) diff --git a/botx/models/system_events/chat_created.py b/botx/models/system_events/chat_created.py new file mode 100644 index 00000000..3e0c08d1 --- /dev/null +++ b/botx/models/system_events/chat_created.py @@ -0,0 +1,115 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Literal, Optional +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import ( + APIChatTypes, + APIUserKinds, + BotAPICommandTypes, + UserKinds, + convert_chat_type_to_domain, + convert_user_kind_to_domain, +) + + +@dataclass +class ChatCreatedMember: + """ChatCreatedEvent member. + + Attributes: + is_admin: Is user admin. + huid: User huid. + username: Username. + kind: User type. + """ + + is_admin: bool + huid: UUID + username: Optional[str] + kind: UserKinds + + +@dataclass +class ChatCreatedEvent(BotCommandBase): + """Event `system:chat_created`. + + Attributes: + sync_id: Event sync id. + chat_id: Created chat id. + chat_name: Created chat name. + chat_type: Created chat type. + host: Created chat cts host. + creator_id: Creator huid. + members: List of created chat members. + """ + + chat: Chat + sync_id: UUID + chat_name: str + creator_id: UUID + members: List[ChatCreatedMember] + + +class BotAPIChatMember(VerifiedPayloadBaseModel): + is_admin: bool = Field(..., alias="admin") + huid: UUID + name: Optional[str] + user_kind: APIUserKinds + + +class BotAPIChatCreatedData(VerifiedPayloadBaseModel): + chat_type: APIChatTypes + creator: UUID + group_chat_id: UUID + members: List[BotAPIChatMember] + name: str + + +class BotAPIChatCreatedPayload(VerifiedPayloadBaseModel): + body: Literal["system:chat_created"] = "system:chat_created" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPIChatCreatedData + + +class BotAPIChatCreated(BotAPIBaseCommand): + payload: BotAPIChatCreatedPayload = Field(..., alias="command") + sender: BotAPIChatContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> ChatCreatedEvent: + members = [ + ChatCreatedMember( + is_admin=member.is_admin, + huid=member.huid, + username=member.name, + kind=convert_user_kind_to_domain(member.user_kind), + ) + for member in self.payload.data.members + ] + + chat = Chat( + id=self.payload.data.group_chat_id, + type=convert_chat_type_to_domain(self.payload.data.chat_type), + ) + + return ChatCreatedEvent( + sync_id=self.sync_id, + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + chat=chat, + chat_name=self.payload.data.name, + creator_id=self.payload.data.creator, + members=members, + raw_command=raw_command, + ) diff --git a/botx/models/system_events/cts_login.py b/botx/models/system_events/cts_login.py new file mode 100644 index 00000000..9cda8b9d --- /dev/null +++ b/botx/models/system_events/cts_login.py @@ -0,0 +1,50 @@ +from dataclasses import dataclass +from typing import Any, Dict, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BaseBotAPIContext, + BotAPIBaseCommand, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.enums import BotAPICommandTypes + + +@dataclass +class CTSLoginEvent(BotCommandBase): + """Event `system:cts_login`. + + Attributes: + huid: user ID. + """ + + huid: UUID + + +class BotAPICTSLoginData(VerifiedPayloadBaseModel): + user_huid: UUID + + +class BotAPICTSLoginPayload(VerifiedPayloadBaseModel): + body: Literal["system:cts_login"] = "system:cts_login" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPICTSLoginData + + +class BotAPICTSLogin(BotAPIBaseCommand): + payload: BotAPICTSLoginPayload = Field(..., alias="command") + sender: BaseBotAPIContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> CTSLoginEvent: + return CTSLoginEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + huid=self.payload.data.user_huid, + ) diff --git a/botx/models/system_events/cts_logout.py b/botx/models/system_events/cts_logout.py new file mode 100644 index 00000000..df1f760e --- /dev/null +++ b/botx/models/system_events/cts_logout.py @@ -0,0 +1,50 @@ +from dataclasses import dataclass +from typing import Any, Dict, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BaseBotAPIContext, + BotAPIBaseCommand, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.enums import BotAPICommandTypes + + +@dataclass +class CTSLogoutEvent(BotCommandBase): + """Event `system:cts_logout`. + + Attributes: + huid: user ID. + """ + + huid: UUID + + +class BotAPICTSLogoutData(VerifiedPayloadBaseModel): + user_huid: UUID + + +class BotAPICTSLogoutPayload(VerifiedPayloadBaseModel): + body: Literal["system:cts_logout"] = "system:cts_logout" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPICTSLogoutData + + +class BotAPICTSLogout(BotAPIBaseCommand): + payload: BotAPICTSLogoutPayload = Field(..., alias="command") + sender: BaseBotAPIContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> CTSLogoutEvent: + return CTSLogoutEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + huid=self.payload.data.user_huid, + ) diff --git a/botx/models/system_events/deleted_from_chat.py b/botx/models/system_events/deleted_from_chat.py new file mode 100644 index 00000000..c5c1c808 --- /dev/null +++ b/botx/models/system_events/deleted_from_chat.py @@ -0,0 +1,57 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import BotAPICommandTypes, convert_chat_type_to_domain + + +@dataclass +class DeletedFromChatEvent(BotCommandBase): + """Event `system:deleted_from_chat`. + + Attributes: + huids: List of deleted from chat user huids. + chat_id: Chat where the user was deleted from. + """ + + huids: List[UUID] + chat: Chat + + +class BotAPIDeletedFromChatData(VerifiedPayloadBaseModel): + deleted_members: List[UUID] + + +class BotAPIDeletedFromChatPayload(VerifiedPayloadBaseModel): + body: Literal["system:deleted_from_chat"] = "system:deleted_from_chat" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPIDeletedFromChatData + + +class BotAPIDeletedFromChat(BotAPIBaseCommand): + payload: BotAPIDeletedFromChatPayload = Field(..., alias="command") + sender: BotAPIChatContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> DeletedFromChatEvent: + return DeletedFromChatEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + huids=self.payload.data.deleted_members, + chat=Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ), + ) diff --git a/botx/models/system_events/internal_bot_notification.py b/botx/models/system_events/internal_bot_notification.py new file mode 100644 index 00000000..17d139aa --- /dev/null +++ b/botx/models/system_events/internal_bot_notification.py @@ -0,0 +1,73 @@ +from dataclasses import dataclass +from typing import Any, Dict, Literal + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotAPIUserContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.bot_sender import BotSender +from botx.models.chats import Chat +from botx.models.enums import BotAPICommandTypes, convert_chat_type_to_domain + + +@dataclass +class InternalBotNotificationEvent(BotCommandBase): + """Event `system:internal_bot_notification`. + + Attributes: + data: user data. + opts: request options. + """ + + data: Dict[str, Any] + opts: Dict[str, Any] + chat: Chat + sender: BotSender + + +class BotAPIInternalBotNotificationData(VerifiedPayloadBaseModel): + data: Dict[str, Any] + opts: Dict[str, Any] + + +class BotAPIInternalBotNotificationPayload(VerifiedPayloadBaseModel): + body: Literal[ + "system:internal_bot_notification" + ] = "system:internal_bot_notification" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPIInternalBotNotificationData + + +class BotAPIBotContext(BotAPIChatContext, BotAPIUserContext): + """Bot context.""" + + +class BotAPIInternalBotNotification(BotAPIBaseCommand): + payload: BotAPIInternalBotNotificationPayload = Field(..., alias="command") + sender: BotAPIBotContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> InternalBotNotificationEvent: + return InternalBotNotificationEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + data=self.payload.data.data, + opts=self.payload.data.opts, + chat=Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ), + sender=BotSender( + huid=self.sender.user_huid, + is_chat_admin=self.sender.is_admin, + is_chat_creator=self.sender.is_creator, + ), + ) diff --git a/botx/models/system_events/left_from_chat.py b/botx/models/system_events/left_from_chat.py new file mode 100644 index 00000000..0be7cf8e --- /dev/null +++ b/botx/models/system_events/left_from_chat.py @@ -0,0 +1,56 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import BotAPICommandTypes, convert_chat_type_to_domain + + +@dataclass +class LeftFromChatEvent(BotCommandBase): + """Event `system:left_from_chat`. + + Attributes: + huids: List of left from chat user huids. + """ + + huids: List[UUID] + chat: Chat + + +class BotAPILeftFromChatData(VerifiedPayloadBaseModel): + left_members: List[UUID] + + +class BotAPILeftFromChatPayload(VerifiedPayloadBaseModel): + body: Literal["system:left_from_chat"] = "system:left_from_chat" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPILeftFromChatData + + +class BotAPILeftFromChat(BotAPIBaseCommand): + payload: BotAPILeftFromChatPayload = Field(..., alias="command") + sender: BotAPIChatContext = Field(..., alias="from") + + def to_domain(self, raw_command: Dict[str, Any]) -> LeftFromChatEvent: + return LeftFromChatEvent( + bot=BotAccount( + id=self.bot_id, + host=self.sender.host, + ), + raw_command=raw_command, + huids=self.payload.data.left_members, + chat=Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ), + ) diff --git a/botx/models/system_events/smartapp_event.py b/botx/models/system_events/smartapp_event.py new file mode 100644 index 00000000..a4c8846c --- /dev/null +++ b/botx/models/system_events/smartapp_event.py @@ -0,0 +1,119 @@ +from dataclasses import dataclass +from typing import Any, Dict, List, Literal +from uuid import UUID + +from pydantic import Field + +from botx.models.api_base import VerifiedPayloadBaseModel +from botx.models.async_files import APIAsyncFile, File, convert_async_file_to_domain +from botx.models.base_command import ( + BotAPIBaseCommand, + BotAPIChatContext, + BotAPIDeviceContext, + BotAPIUserContext, + BotCommandBase, +) +from botx.models.bot_account import BotAccount +from botx.models.chats import Chat +from botx.models.enums import ( + BotAPICommandTypes, + convert_chat_type_to_domain, + convert_client_platform_to_domain, +) +from botx.models.message.incoming_message import UserDevice, UserSender + + +@dataclass +class SmartAppEvent(BotCommandBase): + """Event `system:smartapp_event`. + + Attributes: + ref: Unique request id. + smartapp_id: also personnel chat_id. + data: Payload. + opts: Request options. + smartapp_api_version: Protocol version. + sender: Event sender. + """ + + ref: UUID + smartapp_id: UUID + data: Dict[str, Any] # noqa: WPS110 + opts: Dict[str, Any] + smartapp_api_version: int + files: List[File] + chat: Chat + sender: UserSender + + +class BotAPISmartAppData(VerifiedPayloadBaseModel): + ref: UUID + smartapp_id: UUID + data: Dict[str, Any] # noqa: WPS110 + opts: Dict[str, Any] + smartapp_api_version: int + + +class BotAPISmartAppPayload(VerifiedPayloadBaseModel): + body: Literal["system:smartapp_event"] = "system:smartapp_event" + command_type: Literal[BotAPICommandTypes.SYSTEM] + data: BotAPISmartAppData + metadata: Dict[str, Any] + + +class BotAPISmartAppEventContext( + BotAPIUserContext, + BotAPIChatContext, + BotAPIDeviceContext, +): + """Class for merging contexts.""" + + +class BotAPISmartAppEvent(BotAPIBaseCommand): + payload: BotAPISmartAppPayload = Field(..., alias="command") + sender: BotAPISmartAppEventContext = Field(..., alias="from") + async_files: List[APIAsyncFile] + + def to_domain(self, raw_command: Dict[str, Any]) -> SmartAppEvent: + device = UserDevice( + manufacturer=self.sender.manufacturer, + device_name=self.sender.device, + os=self.sender.device_software, + pushes=None, + timezone=None, + permissions=None, + platform=( + convert_client_platform_to_domain(self.sender.platform) + if self.sender.platform + else None + ), + platform_package_id=self.sender.platform_package_id, + app_version=self.sender.app_version, + locale=self.sender.locale, + ) + + sender = UserSender( + huid=self.sender.user_huid, + ad_login=self.sender.ad_login, + ad_domain=self.sender.ad_domain, + username=self.sender.username, + is_chat_admin=self.sender.is_admin, + is_chat_creator=self.sender.is_creator, + device=device, + ) + + return SmartAppEvent( + bot=BotAccount(id=self.bot_id, host=self.sender.host), + raw_command=raw_command, + ref=self.payload.data.ref, + smartapp_id=self.payload.data.smartapp_id, + data=self.payload.data.data, + opts=self.payload.data.opts, + smartapp_api_version=self.payload.data.smartapp_api_version, + files=[convert_async_file_to_domain(file) for file in self.async_files], + chat=Chat( + id=self.sender.group_chat_id, + type=convert_chat_type_to_domain(self.sender.chat_type), + ), + sender=sender, + ) diff --git a/botx/models/typing.py b/botx/models/typing.py deleted file mode 100644 index ea1f213a..00000000 --- a/botx/models/typing.py +++ /dev/null @@ -1,19 +0,0 @@ -"""Aliases for complex types from `typing` for models.""" - -from typing import List, Union -from uuid import UUID - -from botx.models.buttons import BubbleElement, KeyboardElement - -try: - from typing import Literal # noqa: WPS433 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 - -BubblesRow = List[BubbleElement] -BubbleMarkup = List[BubblesRow] - -KeyboardRow = List[KeyboardElement] -KeyboardMarkup = List[KeyboardRow] - -AvailableRecipients = Union[List[UUID], Literal["all"]] diff --git a/botx/models/users.py b/botx/models/users.py index bfc1c1c5..fa43e7da 100644 --- a/botx/models/users.py +++ b/botx/models/users.py @@ -1,64 +1,28 @@ -"""Entities for users.""" - +from dataclasses import dataclass from typing import List, Optional from uuid import UUID -from botx.models.base import BotXBaseModel -from botx.models.enums import UserKinds +@dataclass +class UserFromSearch: + """User from search. -class UserInChatCreated(BotXBaseModel): - """User that can be included in data in `system:chat_created` event.""" + Attributes: + huid: User huid. + ad_login: User AD login. + ad_domain: User AD domain. + username: User name. + company: User company. + company_position: User company position. + department: User department. + emails: User emails. + """ - #: user HUID. huid: UUID - - #: type of user. - user_kind: UserKinds - - #: user username. - name: Optional[str] - - #: is user administrator in chat. - admin: bool - - -class UserFromSearch(BotXBaseModel): - """User from search request.""" - - #: HUID of user from search. - user_huid: UUID - - #: AD login of user. ad_login: Optional[str] - - # AD domain of user. ad_domain: Optional[str] - - #: visible username. - name: str - - #: user's company. + username: str company: Optional[str] - - #: user's position. company_position: Optional[str] - - #: user's department. department: Optional[str] - - #: user's emails. emails: List[str] - - -class UserFromChatSearch(BotXBaseModel): - """User from chat search request.""" - - #: is user admin of chat. - admin: bool - - #: HUID of user. - user_huid: UUID - - #: type of user. - user_kind: UserKinds diff --git a/botx/shared.py b/botx/shared.py deleted file mode 100644 index 1a07cda1..00000000 --- a/botx/shared.py +++ /dev/null @@ -1,9 +0,0 @@ -"""Shared config for pydantic dataclasses.""" - -from pydantic import BaseConfig - - -class BotXDataclassConfig(BaseConfig): - """Config for pydantic dataclasses that allows custom types.""" - - arbitrary_types_allowed = True diff --git a/botx/testing/__init__.py b/botx/testing/__init__.py deleted file mode 100644 index 47482079..00000000 --- a/botx/testing/__init__.py +++ /dev/null @@ -1,10 +0,0 @@ -"""Definition of entities for using in tests.""" - -from botx.testing.building.builder import MessageBuilder - -try: - from botx.testing.testing_client.client import TestClient # noqa: WPS433 -except ImportError: - TestClient = None # type: ignore # noqa: WPS440 - -__all__ = ("TestClient", "MessageBuilder") # noqa: WPS410 diff --git a/botx/testing/botx_mock/__init__.py b/botx/testing/botx_mock/__init__.py deleted file mode 100644 index 69127fa2..00000000 --- a/botx/testing/botx_mock/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Mock for BotX API for httpx client using Starlette.""" diff --git a/botx/testing/botx_mock/asgi/__init__.py b/botx/testing/botx_mock/asgi/__init__.py deleted file mode 100644 index 01768527..00000000 --- a/botx/testing/botx_mock/asgi/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""ASGI mock for BotX API for using with httpx.""" diff --git a/botx/testing/botx_mock/asgi/application.py b/botx/testing/botx_mock/asgi/application.py deleted file mode 100644 index 836d730d..00000000 --- a/botx/testing/botx_mock/asgi/application.py +++ /dev/null @@ -1,105 +0,0 @@ -"""Definition of Starlette application that is mock for BotX API.""" - -from typing import Any, Dict, List, Sequence, Tuple, Type - -from starlette.applications import Starlette -from starlette.middleware.base import RequestResponseEndpoint -from starlette.routing import Route - -from botx.clients.methods.base import BotXMethod -from botx.testing.botx_mock.asgi.errors import ErrorMiddleware -from botx.testing.botx_mock.asgi.routes import bots # noqa: WPS235 -from botx.testing.botx_mock.asgi.routes import ( - chats, - command, - events, - files, - notification, - notifications, - smartapps, - stickers, - users, -) -from botx.testing.typing import APIMessage, APIRequest - -_ENDPOINTS: Tuple[RequestResponseEndpoint, ...] = ( - # V2 - # bots - bots.get_token, - # V3 - # chats - chats.get_info, - chats.get_bot_chats, - chats.post_add_admin_role, - chats.post_add_user, - chats.post_remove_user, - chats.post_stealth_set, - chats.post_stealth_disable, - chats.post_create, - chats.post_pin_message, - chats.post_unpin_message, - # command - command.post_command_result, - # events - events.post_edit_event, - events.post_reply_event, - # notification - notification.post_notification, - notification.post_notification_direct, - # users - users.get_by_huid, - users.get_by_email, - users.get_by_login, - # notifications - notifications.post_internal_bot_notification, - # files - files.upload_file, - files.download_file, - # stickers - stickers.get_sticker_pack_list, - stickers.get_sticker_pack, - stickers.get_sticker_from_sticker_pack, - stickers.post_add_sticker_into_sticker_pack, - stickers.post_delete_sticker_pack, - stickers.post_delete_sticker_from_sticker_pack, - stickers.post_create_sticker_pack, - stickers.post_edit_sticker_pack, - # smartapps - smartapps.post_smartapp_event, - smartapps.post_smartapp_notification, -) - - -def _create_starlette_routes() -> Sequence[Route]: - routes = [] - - for endpoint in _ENDPOINTS: - url = endpoint.method.__url__ # type: ignore # noqa: WPS609 - method = endpoint.method.__method__ # type: ignore # noqa: WPS609 - routes.append(Route(url, endpoint, methods=[method])) - - return routes - - -def get_botx_asgi_api( - messages: List[APIMessage], - requests: List[APIRequest], - errors: Dict[Type[BotXMethod], Tuple[int, Any]], -) -> Starlette: - """Generate BotX API mock. - - Arguments: - messages: list of message that were sent from bot and should be extended. - requests: all requests that were sent from bot. - errors: errors to be generated by mocked API. - - Returns: - Generated BotX API mock for using with httpx. - """ - botx_app = Starlette(routes=list(_create_starlette_routes())) - botx_app.add_middleware(ErrorMiddleware) - botx_app.state.messages = messages - botx_app.state.requests = requests - botx_app.state.errors = errors - - return botx_app diff --git a/botx/testing/botx_mock/asgi/errors.py b/botx/testing/botx_mock/asgi/errors.py deleted file mode 100644 index d0a13dd3..00000000 --- a/botx/testing/botx_mock/asgi/errors.py +++ /dev/null @@ -1,88 +0,0 @@ -"""Definition of middleware that will generate BotX API errors depending from flag.""" -from typing import Tuple, Type - -from pydantic import BaseModel -from starlette.middleware.base import BaseHTTPMiddleware, RequestResponseEndpoint -from starlette.requests import Request -from starlette.responses import Response -from starlette.routing import Match - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.testing.botx_mock.asgi.responses import PydanticResponse - - -def _fill_request_scope(request: Request) -> None: - routes = request.app.router.routes - for route in routes: - match, scope = route.matches(request) - if match == Match.FULL: - request.scope = {**request.scope, **scope} - - -def _get_error_from_request( - request: Request, -) -> Tuple[Type[BotXMethod], Tuple[int, BaseModel]]: - _fill_request_scope(request) - endpoint = request.scope["endpoint"] - method = endpoint.method - return method, request.app.state.errors.get(method) - - -def should_generate_error_response(request: Request) -> bool: - """Check if mocked API should generate error response. - - Arguments: - request: request from Starlette route that contains application with required - state. - - Returns: - Result of check. - """ - _, status_and_error_to_raise = _get_error_from_request(request) - return bool(status_and_error_to_raise) - - -def generate_error_response(request: Request) -> Response: - """Generate error response for mocked BotX API. - - Arguments: - request: request from Starlette route that contains application with required - state. - - Returns: - Generated response. - """ - method, response_info = _get_error_from_request(request) - status_code, error_data = response_info - - return PydanticResponse( - APIErrorResponse[BaseModel]( - errors=["error from mock"], - reason="asked_for_error", - error_data=error_data, - ), - status_code=status_code, - ) - - -class ErrorMiddleware(BaseHTTPMiddleware): - """Middleware that will generate error response.""" - - async def dispatch( - self, - request: Request, - call_next: RequestResponseEndpoint, - ) -> Response: - """Generate error response for API call or pass request to mocked endpoint. - - Arguments: - request: request that should be handled. - call_next: next executor for mock. - - Returns: - Mocked response. - """ - if should_generate_error_response(request): - return generate_error_response(request) - - return await call_next(request) diff --git a/botx/testing/botx_mock/asgi/messages.py b/botx/testing/botx_mock/asgi/messages.py deleted file mode 100644 index 7ad26966..00000000 --- a/botx/testing/botx_mock/asgi/messages.py +++ /dev/null @@ -1,33 +0,0 @@ -"""Logic for extending messages and requests collections from test client.""" - -import contextlib - -from starlette.requests import Request - -from botx.clients.methods.base import BotXMethod - - -def add_message_to_collection(request: Request, message: BotXMethod) -> None: - """Add new message to messages collection. - - Arguments: - request: request from Starlette endpoint. - message: message that should be added. - """ - app = request.app - with contextlib.suppress(AttributeError): - app.state.messages.append(message) - - add_request_to_collection(request, message) - - -def add_request_to_collection(http_request: Request, api_request: BotXMethod) -> None: - """Add new API request to requests collection. - - Arguments: - http_request: request from Starlette endpoint. - api_request: API request that should be added. - """ - app = http_request.app - with contextlib.suppress(AttributeError): - app.state.requests.append(api_request) diff --git a/botx/testing/botx_mock/asgi/responses.py b/botx/testing/botx_mock/asgi/responses.py deleted file mode 100644 index 11a84d35..00000000 --- a/botx/testing/botx_mock/asgi/responses.py +++ /dev/null @@ -1,82 +0,0 @@ -"""Common responses for mocks.""" - -import uuid -from typing import Any, Optional, Union - -from pydantic import BaseModel -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.clients.types.response_results import ( - InternalBotNotificationResult, - PushResult, -) - - -class PydanticResponse(Response): - """Custom response to encode pydantic model from route.""" - - def __init__( # noqa: WPS211 - self, - model: Optional[BaseModel], - raw_data: Optional[bytes] = None, - status_code: int = 200, - media_type: str = "application/json", - **kwargs: Any, - ) -> None: - """Init custom response. - - Arguments: - model: pydantic model that should be encoded. - raw_data: binary data. - status_code: response HTTP status code. - media_type: content type of response. - kwargs: other arguments to response constructor from starlette. - """ - super().__init__( - raw_data or model.json(by_alias=True), # type: ignore - status_code, - media_type=media_type, - **kwargs, - ) - - -def generate_push_response( - payload: Union[CommandResult, NotificationDirect], -) -> Response: - """Generate response as like new message from bot was pushed. - - Arguments: - payload: pushed message. - - Returns: - Response with sync_id for new message. - """ - sync_id = payload.event_sync_id or uuid.uuid4() - return PydanticResponse( - APIResponse[PushResult](result=PushResult(sync_id=sync_id)), - ) - - -def generate_internal_bot_notification_response( - payload: InternalBotNotification, -) -> Response: - """Generate response as like internal bot notification was sent. - - Arguments: - payload: sent notification. - - Returns: - Response with sync_id for new message. - """ - sync_id = uuid.uuid4() - return PydanticResponse( - APIResponse[InternalBotNotificationResult]( - result=InternalBotNotificationResult(sync_id=sync_id), - ), - ) diff --git a/botx/testing/botx_mock/asgi/routes/__init__.py b/botx/testing/botx_mock/asgi/routes/__init__.py deleted file mode 100644 index c259761e..00000000 --- a/botx/testing/botx_mock/asgi/routes/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition of routes for BotX API mock.""" diff --git a/botx/testing/botx_mock/asgi/routes/bots.py b/botx/testing/botx_mock/asgi/routes/bots.py deleted file mode 100644 index 959ef5b0..00000000 --- a/botx/testing/botx_mock/asgi/routes/bots.py +++ /dev/null @@ -1,26 +0,0 @@ -"""Endpoints for bots resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v2.bots.token import Token -from botx.testing.botx_mock.asgi.messages import add_request_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(Token) -async def get_token(request: Request) -> Response: - """Handle retrieving token from BotX API request. - - Arguments: - request: starlette request for route. - - Returns: - Return response with new token. - """ - request_data = {**request.path_params, **request.query_params} - payload = Token(**request_data) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="real token")) diff --git a/botx/testing/botx_mock/asgi/routes/chats.py b/botx/testing/botx_mock/asgi/routes/chats.py deleted file mode 100644 index 05fd03a5..00000000 --- a/botx/testing/botx_mock/asgi/routes/chats.py +++ /dev/null @@ -1,214 +0,0 @@ -"""Endpoints for chats resource.""" -import uuid -from datetime import datetime as dt - -from starlette import requests, responses - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.chats import add_admin_role # noqa: WPS235 -from botx.clients.methods.v3.chats import ( - add_user, - chat_list, - create, - info, - pin_message, - remove_user, - stealth_disable, - stealth_set, - unpin_message, -) -from botx.clients.types.response_results import ChatCreatedResult -from botx.models import chats, enums, users -from botx.testing.botx_mock.asgi.messages import add_request_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(info.Info) -async def get_info(request: requests.Request) -> responses.Response: - """Handle retrieving information of chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information of chat. - """ - payload = info.Info.parse_obj(request.query_params) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - return PydanticResponse( - APIResponse[chats.ChatFromSearch]( - result=chats.ChatFromSearch( - name="chat name", - chat_type=enums.ChatTypes.group_chat, - creator=uuid.uuid4(), - group_chat_id=payload.group_chat_id, - members=[ - users.UserFromChatSearch( - user_huid=uuid.uuid4(), - user_kind=enums.UserKinds.user, - admin=True, - ), - ], - inserted_at=inserted_at, - ), - ), - ) - - -@bind_implementation_to_method(chat_list.ChatList) -async def get_bot_chats(request: requests.Request) -> responses.Response: - """Return list of bot chats. - - Arguments: - request: HTTP request from Starlette. - - Returns: - List of bot chats. - """ - payload = chat_list.ChatList.parse_obj(request.query_params) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - updated_at = dt.fromisoformat("2019-09-29T10:30:48.358586+00:00") - return PydanticResponse( - APIResponse[chats.BotChatList]( - result=chats.BotChatList( - __root__=[ - chats.BotChatFromList( - name="chat name", - description="test", - chat_type=enums.ChatTypes.group_chat, - group_chat_id=uuid.uuid4(), - members=[uuid.uuid4()], - inserted_at=inserted_at, - updated_at=updated_at, - ), - ], - ), - ), - ) - - -@bind_implementation_to_method(add_user.AddUser) -async def post_add_user(request: requests.Request) -> responses.Response: - """Handle adding of user to chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of adding. - """ - payload = add_user.AddUser.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(remove_user.RemoveUser) -async def post_remove_user(request: requests.Request) -> responses.Response: - """Handle removing of user to chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of removing. - """ - payload = remove_user.RemoveUser.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(stealth_set.StealthSet) -async def post_stealth_set(request: requests.Request) -> responses.Response: - """Handle stealth enabling in chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of enabling stealth. - """ - payload = stealth_set.StealthSet.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(stealth_disable.StealthDisable) -async def post_stealth_disable(request: requests.Request) -> responses.Response: - """Handle stealth disabling in chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of disabling stealth. - """ - payload = stealth_disable.StealthDisable.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(create.Create) -async def post_create(request: requests.Request) -> responses.Response: - """Handle creation of new chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of creation. - """ - payload = create.Create.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse( - APIResponse[ChatCreatedResult](result=ChatCreatedResult(chat_id=uuid.uuid4())), - ) - - -@bind_implementation_to_method(add_admin_role.AddAdminRole) -async def post_add_admin_role(request: requests.Request) -> responses.Response: - """Handle promoting users to admins request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of adding. - """ - payload = add_admin_role.AddAdminRole.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(pin_message.PinMessage) -async def post_pin_message(request: requests.Request) -> responses.Response: - """Handle pinning message in chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of pinning. - """ - payload = pin_message.PinMessage.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="pinned")) - - -@bind_implementation_to_method(unpin_message.UnpinMessage) -async def post_unpin_message(request: requests.Request) -> responses.Response: - """Handle unpinning message in chat request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of unpinning. - """ - payload = unpin_message.UnpinMessage.parse_obj(await request.json()) - add_request_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="unpinned")) diff --git a/botx/testing/botx_mock/asgi/routes/command.py b/botx/testing/botx_mock/asgi/routes/command.py deleted file mode 100644 index bde1af0c..00000000 --- a/botx/testing/botx_mock/asgi/routes/command.py +++ /dev/null @@ -1,24 +0,0 @@ -"""Endpoints for command resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.testing.botx_mock.asgi.messages import add_message_to_collection -from botx.testing.botx_mock.asgi.responses import generate_push_response -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(CommandResult) -async def post_command_result(request: Request) -> Response: - """Handle command result request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = CommandResult.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return generate_push_response(payload) diff --git a/botx/testing/botx_mock/asgi/routes/events.py b/botx/testing/botx_mock/asgi/routes/events.py deleted file mode 100644 index bbe32640..00000000 --- a/botx/testing/botx_mock/asgi/routes/events.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Endpoints for events resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.events.edit_event import EditEvent -from botx.clients.methods.v3.events.reply_event import ReplyEvent -from botx.testing.botx_mock.asgi.messages import add_message_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(EditEvent) -async def post_edit_event(request: Request) -> Response: - """Handle edition of event request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Empty json response. - """ - payload = EditEvent.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="update_pushed")) - - -@bind_implementation_to_method(ReplyEvent) -async def post_reply_event(request: Request) -> Response: - """Handle reply event request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Empty json response. - """ - payload = ReplyEvent.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="reply_pushed")) diff --git a/botx/testing/botx_mock/asgi/routes/files.py b/botx/testing/botx_mock/asgi/routes/files.py deleted file mode 100644 index 37c4f56e..00000000 --- a/botx/testing/botx_mock/asgi/routes/files.py +++ /dev/null @@ -1,60 +0,0 @@ -"""Endpoints for chats resource.""" - -import json - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.files.download import DownloadFile -from botx.clients.methods.v3.files.upload import UploadFile -from botx.models.files import File, MetaFile -from botx.testing.botx_mock.asgi.messages import add_request_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.entities import create_test_metafile - - -@bind_implementation_to_method(UploadFile) -async def upload_file(request: Request) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with metadata of file. - """ - form = dict(await request.form()) - meta = json.loads(form["meta"]) - filename = form["content"].filename # type: ignore - file = File.from_file( - filename=filename, - file=form["content"].file, # type: ignore - ) - payload = UploadFile( - group_chat_id=form["group_chat_id"], # type: ignore - file=file, - meta=meta, - ) - add_request_to_collection(request, payload) - return PydanticResponse( - APIResponse[MetaFile]( - result=create_test_metafile(filename), - ), - ) - - -@bind_implementation_to_method(DownloadFile) -async def download_file(request: Request) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with file content. - """ - payload = DownloadFile.parse_obj(request.query_params) - add_request_to_collection(request, payload) - return PydanticResponse(model=None, raw_data=b"content", media_type="text/plain") diff --git a/botx/testing/botx_mock/asgi/routes/notification.py b/botx/testing/botx_mock/asgi/routes/notification.py deleted file mode 100644 index 6b55e604..00000000 --- a/botx/testing/botx_mock/asgi/routes/notification.py +++ /dev/null @@ -1,44 +0,0 @@ -"""Endpoints for notification resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v3.notification.notification import Notification -from botx.testing.botx_mock.asgi.messages import add_message_to_collection -from botx.testing.botx_mock.asgi.responses import ( - PydanticResponse, - generate_push_response, -) -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(Notification) -async def post_notification(request: Request) -> Response: - """Handle pushed notification request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = Notification.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="notification_pushed")) - - -@bind_implementation_to_method(NotificationDirect) -async def post_notification_direct(request: Request) -> Response: - """Handle pushed notification request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = NotificationDirect.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return generate_push_response(payload) diff --git a/botx/testing/botx_mock/asgi/routes/notifications.py b/botx/testing/botx_mock/asgi/routes/notifications.py deleted file mode 100644 index 0e71827c..00000000 --- a/botx/testing/botx_mock/asgi/routes/notifications.py +++ /dev/null @@ -1,28 +0,0 @@ -"""Endpoints for notification resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.testing.botx_mock.asgi.messages import add_message_to_collection -from botx.testing.botx_mock.asgi.responses import ( - generate_internal_bot_notification_response, -) -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(InternalBotNotification) -async def post_internal_bot_notification(request: Request) -> Response: - """Handle pushed notification request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = InternalBotNotification.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return generate_internal_bot_notification_response(payload) diff --git a/botx/testing/botx_mock/asgi/routes/smartapps.py b/botx/testing/botx_mock/asgi/routes/smartapps.py deleted file mode 100644 index 94201969..00000000 --- a/botx/testing/botx_mock/asgi/routes/smartapps.py +++ /dev/null @@ -1,41 +0,0 @@ -"""Endpoints for smartapps.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.smartapps.smartapp_event import SmartAppEvent -from botx.clients.methods.v3.smartapps.smartapp_notification import SmartAppNotification -from botx.testing.botx_mock.asgi.messages import add_message_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method - - -@bind_implementation_to_method(SmartAppEvent) -async def post_smartapp_event(request: Request) -> Response: - """Handle pushed smartapp event request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = SmartAppEvent.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="smartapp_event_pushed")) - - -@bind_implementation_to_method(SmartAppNotification) -async def post_smartapp_notification(request: Request) -> Response: - """Handle pushed smartapp notification request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with sync_id of pushed message. - """ - payload = SmartAppNotification.parse_obj(await request.json()) - add_message_to_collection(request, payload) - return PydanticResponse(APIResponse[str](result="smartapp_notification_pushed")) diff --git a/botx/testing/botx_mock/asgi/routes/stickers.py b/botx/testing/botx_mock/asgi/routes/stickers.py deleted file mode 100644 index 75ec4033..00000000 --- a/botx/testing/botx_mock/asgi/routes/stickers.py +++ /dev/null @@ -1,287 +0,0 @@ -"""Endpoints for stickers.""" -import uuid -from datetime import datetime as dt - -from starlette import requests, responses - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.stickers import ( - add_sticker, - create_sticker_pack, - delete_sticker, - delete_sticker_pack, - edit_sticker_pack, - sticker, - sticker_pack, - sticker_pack_list, -) -from botx.models import stickers -from botx.testing.botx_mock.asgi.messages import add_request_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.content import PNG_DATA - - -@bind_implementation_to_method(sticker_pack_list.GetStickerPackList) -async def get_sticker_pack_list(request: requests.Request) -> responses.Response: - """Handle retrieving information of sticker pack list request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with list of sticker packs. - """ - payload = sticker_pack_list.GetStickerPackList.parse_obj(request.query_params) - add_request_to_collection(request, payload) - - pagination = stickers.Pagination(after=PNG_DATA) - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - sticker_packs = [ - stickers.StickerPackPreview( - id=uuid.uuid4(), - name="Test sticker pack", - public=False, - stickers_count=1, - inserted_at=inserted_at, - ), - ] - sticker_pack_list_response = stickers.StickerPackList( - packs=sticker_packs, - pagination=pagination, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPackList]( - result=sticker_pack_list_response, - ), - ) - - -@bind_implementation_to_method(sticker_pack.GetStickerPack) -async def get_sticker_pack(request: requests.Request) -> responses.Response: - """Handle retrieving information of sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information of sticker pack. - """ - payload = sticker_pack.GetStickerPack.parse_obj(request.path_params) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - sticker_pack_preview = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_pack_preview, - ), - ) - - -@bind_implementation_to_method(sticker.GetSticker) -async def get_sticker_from_sticker_pack( - request: requests.Request, -) -> responses.Response: - """Handle retrieving information of sticker from sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information of sticker from sticker pack. - """ - payload = sticker.GetSticker.parse_obj(request.path_params) - add_request_to_collection(request, payload) - - sticker_from_sticker_pack = stickers.StickerFromPack( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - preview="http://preview_link.com", - ) - - return PydanticResponse( - APIResponse[stickers.StickerFromPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(add_sticker.AddSticker) -async def post_add_sticker_into_sticker_pack( - request: requests.Request, -) -> responses.Response: - """Handle adding of sticker to sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of adding. - """ - payload = add_sticker.AddSticker.parse_obj(await request.json()) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - sticker_from_sticker_pack = stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ) - - return PydanticResponse( - APIResponse[stickers.Sticker]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(create_sticker_pack.CreateStickerPack) -async def post_create_sticker_pack(request: requests.Request) -> responses.Response: - """Handle creating of sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of creating. - """ - payload = create_sticker_pack.CreateStickerPack.parse_obj(request.query_params) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - sticker_from_sticker_pack = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(edit_sticker_pack.EditStickerPack) -async def post_edit_sticker_pack(request: requests.Request) -> responses.Response: - """Handle editing of sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of editing. - """ - payload = edit_sticker_pack.EditStickerPack.parse_obj(await request.json()) - add_request_to_collection(request, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - sticker_from_sticker_pack = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(delete_sticker_pack.DeleteStickerPack) -async def post_delete_sticker_pack(request: requests.Request) -> responses.Response: - """Handle deleting of sticker pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of deleting. - """ - payload = delete_sticker_pack.DeleteStickerPack.parse_obj(request.path_params) - add_request_to_collection(request, payload) - - return PydanticResponse( - APIResponse[str]( - result="sticker_pack_deleted", - ), - ) - - -@bind_implementation_to_method(delete_sticker.DeleteSticker) -async def post_delete_sticker_from_sticker_pack( - request: requests.Request, -) -> responses.Response: - """Handle deleting of sticker from sticker pack pack request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with result of deleting. - """ - payload = delete_sticker.DeleteSticker.parse_obj(request.path_params) - add_request_to_collection(request, payload) - - return PydanticResponse( - APIResponse[str]( - result="sticker_deleted", - ), - ) diff --git a/botx/testing/botx_mock/asgi/routes/users.py b/botx/testing/botx_mock/asgi/routes/users.py deleted file mode 100644 index fc54fc23..00000000 --- a/botx/testing/botx_mock/asgi/routes/users.py +++ /dev/null @@ -1,69 +0,0 @@ -"""Endpoints for chats resource.""" - -from starlette.requests import Request -from starlette.responses import Response - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.users.by_email import ByEmail -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.clients.methods.v3.users.by_login import ByLogin -from botx.models.users import UserFromSearch -from botx.testing.botx_mock.asgi.messages import add_request_to_collection -from botx.testing.botx_mock.asgi.responses import PydanticResponse -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.entities import create_test_user - - -@bind_implementation_to_method(ByHUID) -async def get_by_huid(request: Request) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information about user. - """ - payload = ByHUID.parse_obj(request.query_params) - add_request_to_collection(request, payload) - return PydanticResponse( - APIResponse[UserFromSearch]( - result=create_test_user(user_huid=payload.user_huid), - ), - ) - - -@bind_implementation_to_method(ByEmail) -async def get_by_email(request: Request) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information about user. - """ - payload = ByEmail.parse_obj(request.query_params) - add_request_to_collection(request, payload) - return PydanticResponse( - APIResponse[UserFromSearch](result=create_test_user(email=payload.email)), - ) - - -@bind_implementation_to_method(ByLogin) -async def get_by_login(request: Request) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Starlette. - - Returns: - Response with information about user. - """ - payload = ByLogin.parse_obj(request.query_params) - add_request_to_collection(request, payload) - return PydanticResponse( - APIResponse[UserFromSearch]( - result=create_test_user(ad=(payload.ad_login, payload.ad_domain)), - ), - ) diff --git a/botx/testing/botx_mock/binders.py b/botx/testing/botx_mock/binders.py deleted file mode 100644 index f20c9a1a..00000000 --- a/botx/testing/botx_mock/binders.py +++ /dev/null @@ -1,21 +0,0 @@ -"""Decorator for binding custom BotXMethod with implementation.""" -from typing import Any, Callable, Type - -from botx.clients.methods.base import BotXMethod - - -def bind_implementation_to_method(method: Type[BotXMethod]) -> Callable[..., Any]: - """Bind implementation of async route to method. - - Arguments: - method: method class to bind. - - Returns: - Decorator that binds method and returns it. - """ - - def decorator(func: Callable[..., Any]) -> Callable[..., Any]: - func.method = method # type: ignore - return func - - return decorator diff --git a/botx/testing/botx_mock/entities.py b/botx/testing/botx_mock/entities.py deleted file mode 100644 index 2ea60768..00000000 --- a/botx/testing/botx_mock/entities.py +++ /dev/null @@ -1,62 +0,0 @@ -"""Predefined mocked entities builder for routes.""" -import uuid -from typing import Optional, Tuple - -from botx.models.enums import AttachmentsTypes -from botx.models.files import MetaFile -from botx.models.users import UserFromSearch - - -def create_test_user( - *, - user_huid: Optional[uuid.UUID] = None, - email: Optional[str] = None, - ad: Optional[Tuple[str, str]] = None, -) -> UserFromSearch: - """Build test user for using in search. - - Arguments: - user_huid: HUID of user for search. - email: email of user for search. - ad: AD credentials of user for search. - - Returns: - "Found" user. - """ - return UserFromSearch( - user_huid=user_huid or uuid.uuid4(), - ad_login=ad[0] if ad else "ad_login", - ad_domain=ad[1] if ad else "ad_domain", - name="test user", - company="test company", - company_position="test position", - department="test department", - emails=[email or "test@example.com"], - ) - - -def create_test_metafile(filename: str = None) -> MetaFile: - """Build test metafile for using in uploading. - - Arguments: - filename: name of uploaded file. - - Returns: - Metadata of uploaded file. - """ - return MetaFile( - type=AttachmentsTypes.image, - file="https://service.to./image", - file_mime_type="image/png", - file_name=filename or "image.png", - file_preview=None, - file_preview_height=None, - file_preview_width=None, - file_size=100, - file_hash="W1Sn1AkotkOpH0", - file_encryption_algo="stream", - chunk_size=10, - file_id=uuid.UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), - caption=None, - duration=None, - ) diff --git a/botx/testing/botx_mock/wsgi/__init__.py b/botx/testing/botx_mock/wsgi/__init__.py deleted file mode 100644 index 7d2452e7..00000000 --- a/botx/testing/botx_mock/wsgi/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""WSGI mock for BotX API for using with httpx.""" diff --git a/botx/testing/botx_mock/wsgi/application.py b/botx/testing/botx_mock/wsgi/application.py deleted file mode 100644 index 91fe6b36..00000000 --- a/botx/testing/botx_mock/wsgi/application.py +++ /dev/null @@ -1,105 +0,0 @@ -"""Definition of Starlette application that is mock for BotX API.""" - -from typing import Any, Callable, Dict, List, Sequence, Tuple, Type - -from molten import App, JSONParser, Route, Settings, SettingsComponent - -from botx.clients.methods.base import BotXMethod -from botx.testing.botx_mock.wsgi.errors import error_middleware -from botx.testing.botx_mock.wsgi.routes import bots # noqa: WPS235 -from botx.testing.botx_mock.wsgi.routes import ( - chats, - command, - events, - files, - notification, - notifications, - smartapps, - stickers, - users, -) -from botx.testing.typing import APIMessage, APIRequest - -_ENDPOINTS: Tuple[Callable[..., Any], ...] = ( - # V2 - # bots - bots.get_token, - # V3 - # chats - chats.get_info, - chats.get_bot_chats, - chats.post_add_admin_role, - chats.post_add_user, - chats.post_remove_user, - chats.post_stealth_set, - chats.post_stealth_disable, - chats.post_create, - chats.post_pin_message, - chats.post_unpin_message, - # command - command.post_command_result, - # events - events.post_edit_event, - # notification - notification.post_notification, - notification.post_notification_direct, - # users - users.get_by_huid, - users.get_by_email, - users.get_by_login, - # notifications - notifications.post_internal_bot_notification, - # files - files.upload_file, - files.download_file, - # stickers - stickers.get_sticker_pack_list, - stickers.get_sticker_pack, - stickers.get_sticker_from_sticker_pack, - stickers.post_add_sticker_into_sticker_pack, - stickers.post_delete_sticker_pack, - stickers.delete_sticker_from_sticker_pack, - stickers.post_create_sticker_pack, - stickers.post_edit_sticker_pack, - # smartapps - smartapps.post_smartapp_event, - smartapps.post_smartapp_notification, -) - - -def _create_molten_routes() -> Sequence[Route]: - routes = [] - - for endpoint in _ENDPOINTS: - url = endpoint.method.__url__ # type: ignore # noqa: WPS609 - method = endpoint.method.__method__ # type: ignore # noqa: WPS609 - routes.append(Route(url, endpoint, method=method)) - - return routes - - -def get_botx_wsgi_api( - messages: List[APIMessage], - requests: List[APIRequest], - errors: Dict[Type[BotXMethod], Tuple[int, Any]], -) -> App: - """Generate BotX API mock. - - Arguments: - messages: list of message that were sent from bot and should be extended. - requests: all requests that were sent from bot. - errors: errors to be generated by mocked API. - - Returns: - Generated BotX API mock for using with httpx. - """ - return App( - components=[ - SettingsComponent( - Settings(messages=messages, requests=requests, errors=errors), - ), - ], - routes=list(_create_molten_routes()), - middleware=[error_middleware], - parsers=[JSONParser()], - ) diff --git a/botx/testing/botx_mock/wsgi/errors.py b/botx/testing/botx_mock/wsgi/errors.py deleted file mode 100644 index 56705771..00000000 --- a/botx/testing/botx_mock/wsgi/errors.py +++ /dev/null @@ -1,86 +0,0 @@ -"""Definition of middleware that will generate BotX API errors depending from flag.""" -from typing import Any, Callable, Tuple, Type, cast - -from molten import BaseApp, Request, Response, Route, Settings -from pydantic import BaseModel - -from botx.clients.methods.base import APIErrorResponse, BotXMethod -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -def _get_error_from_request( - request: Request, - app: BaseApp, - settings: Settings, -) -> Tuple[Type[BotXMethod], Tuple[int, BaseModel]]: - match = app.router.match(request.method, request.path) - route, _ = cast(Tuple[Route, Any], match) - method = route.handler.method # type: ignore - return method, settings["errors"].get(method) - - -def should_generate_error_response( - request: Request, - app: BaseApp, - settings: Settings, -) -> bool: - """Check if mocked API should generate error response. - - Arguments: - request: request from molten route. - app: molten app that serves request. - settings: application settings with storage. - - Returns: - Result of check. - """ - _, status_and_error_to_raise = _get_error_from_request(request, app, settings) - return bool(status_and_error_to_raise) - - -def generate_error_response( - request: Request, - app: BaseApp, - settings: Settings, -) -> Response: - """Generate error response for mocked BotX API. - - Arguments: - request: request from molten route. - app: molten app that serves request. - settings: application settings with storage. - - Returns: - Generated response. - """ - method, response_info = _get_error_from_request(request, app, settings) - status_code, error_data = response_info - - return PydanticResponse( - APIErrorResponse[BaseModel]( - errors=["error from mock"], - reason="asked_for_error", - error_data=error_data, - ), - # TODO: Drop unnecessary description. - status_code="{0} ".format(status_code), - ) - - -def error_middleware(handler: Callable[..., Any]) -> Callable[..., Any]: - """Middleware that will generate error response. - - Arguments: - handler: next handler for request for molten. - - Returns: - Created handler for request. - """ - - def decorator(request: Request, app: BaseApp, settings: Settings) -> Any: - if should_generate_error_response(request, app, settings): - return generate_error_response(request, app, settings) - - return handler() - - return decorator diff --git a/botx/testing/botx_mock/wsgi/messages.py b/botx/testing/botx_mock/wsgi/messages.py deleted file mode 100644 index 4b0403d2..00000000 --- a/botx/testing/botx_mock/wsgi/messages.py +++ /dev/null @@ -1,28 +0,0 @@ -"""Logic for extending messages and requests collections from test client.""" - -from molten import Settings - -from botx.clients.methods.base import BotXMethod - - -def add_message_to_collection(settings: Settings, message: BotXMethod) -> None: - """Add new message to messages collection. - - Arguments: - settings: application settings with storage. - message: message that should be added. - """ - messages = settings["messages"] - messages.append(message) - add_request_to_collection(settings, message) - - -def add_request_to_collection(settings: Settings, api_request: BotXMethod) -> None: - """Add new API request to requests collection. - - Arguments: - settings: application settings with storage. - api_request: API request that should be added. - """ - requests = settings["requests"] - requests.append(api_request) diff --git a/botx/testing/botx_mock/wsgi/responses.py b/botx/testing/botx_mock/wsgi/responses.py deleted file mode 100644 index 30688a58..00000000 --- a/botx/testing/botx_mock/wsgi/responses.py +++ /dev/null @@ -1,81 +0,0 @@ -"""Common responses for mocks.""" - -import uuid -from typing import Any, Dict, Optional, Union - -from molten import HTTP_200, Response -from pydantic import BaseModel - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.clients.types.response_results import ( - InternalBotNotificationResult, - PushResult, -) - - -class PydanticResponse(Response): - """Custom response to encode pydantic model from route.""" - - def __init__( - self, - model: Optional[BaseModel], - raw_data: Optional[str] = None, - status_code: str = HTTP_200, - headers: Optional[Dict[Any, Any]] = None, - ) -> None: - """Init custom response. - - Arguments: - model: pydantic model that should be encoded. - raw_data: raw data that should be encoded. - status_code: response HTTP status code. - headers: headers for response. - """ - headers = headers or {"Content-Type": "application/json"} - - super().__init__( - status_code, - headers, - raw_data or model.json(by_alias=True), # type: ignore - ) - - -def generate_push_response( - payload: Union[CommandResult, NotificationDirect], -) -> Response: - """Generate response as like new message from bot was pushed. - - Arguments: - payload: pushed message. - - Returns: - Response with sync_id for new message. - """ - sync_id = payload.event_sync_id or uuid.uuid4() - return PydanticResponse( - APIResponse[PushResult](result=PushResult(sync_id=sync_id)), - ) - - -def generate_internal_bot_notification_response( - payload: InternalBotNotification, -) -> Response: - """Generate response as like internal bot notification was sent. - - Arguments: - payload: sent notification. - - Returns: - Response with sync_id for new message. - """ - sync_id = uuid.uuid4() - return PydanticResponse( - APIResponse[InternalBotNotificationResult]( - result=InternalBotNotificationResult(sync_id=sync_id), - ), - ) diff --git a/botx/testing/botx_mock/wsgi/routes/__init__.py b/botx/testing/botx_mock/wsgi/routes/__init__.py deleted file mode 100644 index c259761e..00000000 --- a/botx/testing/botx_mock/wsgi/routes/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition of routes for BotX API mock.""" diff --git a/botx/testing/botx_mock/wsgi/routes/bots.py b/botx/testing/botx_mock/wsgi/routes/bots.py deleted file mode 100644 index 87dcc39c..00000000 --- a/botx/testing/botx_mock/wsgi/routes/bots.py +++ /dev/null @@ -1,28 +0,0 @@ -"""Endpoints for bots resource.""" -from uuid import UUID - -from molten import Request, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v2.bots.token import Token -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_request_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(Token) -def get_token(bot_id: str, request: Request, settings: Settings) -> Response: - """Handle retrieving token from BotX API request. - - Arguments: - bot_id: ID of bot from query params. - request: modten request for route. - settings: application settings with storage. - - Returns: - Return response with new token. - """ - signature = request.params["signature"] - payload = Token(bot_id=UUID(bot_id), signature=signature) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="real token")) diff --git a/botx/testing/botx_mock/wsgi/routes/chats.py b/botx/testing/botx_mock/wsgi/routes/chats.py deleted file mode 100644 index f0935f28..00000000 --- a/botx/testing/botx_mock/wsgi/routes/chats.py +++ /dev/null @@ -1,224 +0,0 @@ -"""Endpoints for chats resource.""" -import uuid -from datetime import datetime as dt - -from molten import Request, RequestData, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.chats import add_admin_role # noqa: WPS235 -from botx.clients.methods.v3.chats import ( - add_user, - chat_list, - create, - info, - pin_message, - remove_user, - stealth_disable, - stealth_set, - unpin_message, -) -from botx.clients.types.response_results import ChatCreatedResult -from botx.models import chats, enums, users -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_request_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(info.Info) -def get_info(request: Request, settings: Settings) -> Response: - """Handle retrieving information of chat request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information of chat. - """ - payload = info.Info.parse_obj(request.params) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - return PydanticResponse( - APIResponse[chats.ChatFromSearch]( - result=chats.ChatFromSearch( - name="chat name", - chat_type=enums.ChatTypes.group_chat, - creator=uuid.uuid4(), - group_chat_id=payload.group_chat_id, - members=[ - users.UserFromChatSearch( - user_huid=uuid.uuid4(), - user_kind=enums.UserKinds.user, - admin=True, - ), - ], - inserted_at=inserted_at, - ), - ), - ) - - -@bind_implementation_to_method(chat_list.ChatList) -def get_bot_chats(request: Request, settings: Settings) -> Response: - """Return list of bot chats. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - List of bot chats. - """ - payload = chat_list.ChatList.parse_obj(request.params) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - updated_at = dt.fromisoformat("2019-09-29T10:30:48.358586+00:00") - return PydanticResponse( - APIResponse[chats.BotChatList]( - result=chats.BotChatList( - __root__=[ - chats.BotChatFromList( - name="chat name", - description="test", - chat_type=enums.ChatTypes.group_chat, - group_chat_id=uuid.uuid4(), - members=[uuid.uuid4()], - inserted_at=inserted_at, - updated_at=updated_at, - ), - ], - ), - ), - ) - - -@bind_implementation_to_method(add_user.AddUser) -def post_add_user(request_data: RequestData, settings: Settings) -> Response: - """Handle adding of user to chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of adding. - """ - payload = add_user.AddUser.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(remove_user.RemoveUser) -def post_remove_user(request_data: RequestData, settings: Settings) -> Response: - """Handle removing of user to chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of removing. - """ - payload = remove_user.RemoveUser.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(stealth_set.StealthSet) -def post_stealth_set(request_data: RequestData, settings: Settings) -> Response: - """Handle stealth enabling in chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of enabling stealth. - """ - payload = stealth_set.StealthSet.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(stealth_disable.StealthDisable) -def post_stealth_disable(request_data: RequestData, settings: Settings) -> Response: - """Handle stealth disabling in chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of disabling stealth. - """ - payload = stealth_disable.StealthDisable.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(create.Create) -def post_create(request_data: RequestData, settings: Settings) -> Response: - """Handle creation of new chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of creation. - """ - payload = create.Create.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse( - APIResponse[ChatCreatedResult](result=ChatCreatedResult(chat_id=uuid.uuid4())), - ) - - -@bind_implementation_to_method(add_admin_role.AddAdminRole) -def post_add_admin_role(request_data: RequestData, settings: Settings) -> Response: - """Handle promoting users to admins request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of adding. - """ - payload = add_admin_role.AddAdminRole.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[bool](result=True)) - - -@bind_implementation_to_method(pin_message.PinMessage) -def post_pin_message(request_data: RequestData, settings: Settings) -> Response: - """Handle pinning message in chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of pinning. - """ - payload = pin_message.PinMessage.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="pinned")) - - -@bind_implementation_to_method(unpin_message.UnpinMessage) -def post_unpin_message(request_data: RequestData, settings: Settings) -> Response: - """Handle unpinning message in chat request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of unpinning. - """ - payload = unpin_message.UnpinMessage.parse_obj(request_data) - add_request_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="pinned")) diff --git a/botx/testing/botx_mock/wsgi/routes/command.py b/botx/testing/botx_mock/wsgi/routes/command.py deleted file mode 100644 index 0f018cc3..00000000 --- a/botx/testing/botx_mock/wsgi/routes/command.py +++ /dev/null @@ -1,23 +0,0 @@ -"""Endpoints for command resource.""" -from molten import RequestData, Response, Settings - -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_message_to_collection -from botx.testing.botx_mock.wsgi.responses import generate_push_response - - -@bind_implementation_to_method(CommandResult) -def post_command_result(request_data: RequestData, settings: Settings) -> Response: - """Handle command result request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = CommandResult.parse_obj(request_data) - add_message_to_collection(settings, payload) - return generate_push_response(payload) diff --git a/botx/testing/botx_mock/wsgi/routes/events.py b/botx/testing/botx_mock/wsgi/routes/events.py deleted file mode 100644 index 5a199027..00000000 --- a/botx/testing/botx_mock/wsgi/routes/events.py +++ /dev/null @@ -1,24 +0,0 @@ -"""Endpoints for events resource.""" -from molten import RequestData, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.events.edit_event import EditEvent -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_message_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(EditEvent) -def post_edit_event(request_data: RequestData, settings: Settings) -> Response: - """Handle edition of event request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Empty json response. - """ - payload = EditEvent.parse_obj(request_data) - add_message_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="update_pushed")) diff --git a/botx/testing/botx_mock/wsgi/routes/files.py b/botx/testing/botx_mock/wsgi/routes/files.py deleted file mode 100644 index 12be8c3c..00000000 --- a/botx/testing/botx_mock/wsgi/routes/files.py +++ /dev/null @@ -1,70 +0,0 @@ -"""Endpoints for chats resource.""" - -import json - -from molten import Header, MultiPartParser, Request, RequestInput, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.files.download import DownloadFile -from botx.clients.methods.v3.files.upload import UploadFile -from botx.models.files import File, MetaFile -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.entities import create_test_metafile -from botx.testing.botx_mock.wsgi.messages import add_request_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(UploadFile) -def upload_file(request: Request, settings: Settings) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with metadata of file. - """ - headers = dict(request.headers) - form = MultiPartParser().parse( - Header(headers["content-type"]), - Header(headers["content-length"]), - RequestInput(request.body_file), - ) - - meta = json.loads(form["meta"]) # type: ignore - file = File.from_file( - filename=form["content"].filename, # type: ignore - file=form["content"].stream, # type: ignore - ) - payload = UploadFile( - group_chat_id=form["group_chat_id"], # type: ignore - file=file, - meta=meta, - ) - add_request_to_collection(settings, payload) - return PydanticResponse( - APIResponse[MetaFile]( - result=create_test_metafile(), - ), - ) - - -@bind_implementation_to_method(DownloadFile) -def download_file(request: Request, settings: Settings) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with file content. - """ - payload = DownloadFile.parse_obj(request.params) - add_request_to_collection(settings, payload) - return PydanticResponse( - model=None, - raw_data="content", - headers={"Content-Type": "text/plain"}, - ) diff --git a/botx/testing/botx_mock/wsgi/routes/notification.py b/botx/testing/botx_mock/wsgi/routes/notification.py deleted file mode 100644 index 46dad979..00000000 --- a/botx/testing/botx_mock/wsgi/routes/notification.py +++ /dev/null @@ -1,44 +0,0 @@ -"""Endpoints for notification resource.""" -from molten import RequestData, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v3.notification.notification import Notification -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_message_to_collection -from botx.testing.botx_mock.wsgi.responses import ( - PydanticResponse, - generate_push_response, -) - - -@bind_implementation_to_method(Notification) -def post_notification(request_data: RequestData, settings: Settings) -> Response: - """Handle pushed notification request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = Notification.parse_obj(request_data) - add_message_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="notification_pushed")) - - -@bind_implementation_to_method(NotificationDirect) -def post_notification_direct(request_data: RequestData, settings: Settings) -> Response: - """Handle pushed notification request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = NotificationDirect.parse_obj(request_data) - add_message_to_collection(settings, payload) - return generate_push_response(payload) diff --git a/botx/testing/botx_mock/wsgi/routes/notifications.py b/botx/testing/botx_mock/wsgi/routes/notifications.py deleted file mode 100644 index cf8bda5c..00000000 --- a/botx/testing/botx_mock/wsgi/routes/notifications.py +++ /dev/null @@ -1,30 +0,0 @@ -"""Endpoints for notifications resource.""" -from molten import RequestData, Response, Settings - -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_message_to_collection -from botx.testing.botx_mock.wsgi.responses import ( - generate_internal_bot_notification_response, -) - - -@bind_implementation_to_method(InternalBotNotification) -def post_internal_bot_notification( - request_data: RequestData, - settings: Settings, -) -> Response: - """Handle pushed notification request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = InternalBotNotification.parse_obj(request_data) - add_message_to_collection(settings, payload) - return generate_internal_bot_notification_response(payload) diff --git a/botx/testing/botx_mock/wsgi/routes/smartapps.py b/botx/testing/botx_mock/wsgi/routes/smartapps.py deleted file mode 100644 index 81827dc7..00000000 --- a/botx/testing/botx_mock/wsgi/routes/smartapps.py +++ /dev/null @@ -1,45 +0,0 @@ -"""Endpoints for smartapps.""" - -from molten import RequestData, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.smartapps.smartapp_event import SmartAppEvent -from botx.clients.methods.v3.smartapps.smartapp_notification import SmartAppNotification -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_message_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(SmartAppEvent) -def post_smartapp_event(request_data: RequestData, settings: Settings) -> Response: - """Handle pushed smartapp event request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = SmartAppEvent.parse_obj(request_data) - add_message_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="smartapp_event_pushed")) - - -@bind_implementation_to_method(SmartAppNotification) -def post_smartapp_notification( - request_data: RequestData, - settings: Settings, -) -> Response: - """Handle pushed smartapp notification request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with sync_id of pushed message. - """ - payload = SmartAppNotification.parse_obj(request_data) - add_message_to_collection(settings, payload) - return PydanticResponse(APIResponse[str](result="smartapp_notification_pushed")) diff --git a/botx/testing/botx_mock/wsgi/routes/stickers.py b/botx/testing/botx_mock/wsgi/routes/stickers.py deleted file mode 100644 index 85f984df..00000000 --- a/botx/testing/botx_mock/wsgi/routes/stickers.py +++ /dev/null @@ -1,292 +0,0 @@ -"""Endpoints for stickers.""" -import uuid -from datetime import datetime as dt - -from molten import Request, RequestData, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.stickers import ( - add_sticker, - create_sticker_pack, - delete_sticker, - delete_sticker_pack, - edit_sticker_pack, - sticker, - sticker_pack, - sticker_pack_list, -) -from botx.models import stickers -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.wsgi.messages import add_request_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(sticker_pack_list.GetStickerPackList) -def get_sticker_pack_list(request: Request, settings: Settings) -> Response: - """Handle retrieving information of sticker pack list request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with list of sticker packs. - """ - payload = sticker_pack_list.GetStickerPackList.parse_obj(request.params) - add_request_to_collection(settings, payload) - - pagination = stickers.Pagination( - after="ABAmCAFTpX1ajK8O_ezuPEQ1AA0ACnVwZGF0ZWRfYXQAf____gAAAAA=", - ) - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - sticker_packs = [ - stickers.StickerPackPreview( - id=uuid.uuid4(), - name="Test sticker pack", - public=False, - stickers_count=1, - inserted_at=inserted_at, - ), - ] - pack_list = stickers.StickerPackList( - packs=sticker_packs, - pagination=pagination, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPackList]( - result=pack_list, - ), - ) - - -@bind_implementation_to_method(sticker_pack.GetStickerPack) -def get_sticker_pack(request: Request, settings: Settings) -> Response: - """Handle retrieving information of sticker pack request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information of sticker pack. - """ - payload = sticker_pack.GetStickerPack.parse_obj(request.params) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - sticker_pack_preview = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_pack_preview, - ), - ) - - -@bind_implementation_to_method(sticker.GetSticker) -def get_sticker_from_sticker_pack(request: Request, settings: Settings) -> Response: - """Handle retrieving information of sticker from sticker pack request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information of sticker from sticker pack. - """ - payload = sticker.GetSticker.parse_obj(request.params) - - add_request_to_collection(settings, payload) - sticker_from_sticker_pack = stickers.StickerFromPack( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - preview="http://preview_link.com", - ) - - return PydanticResponse( - APIResponse[stickers.StickerFromPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(add_sticker.AddSticker) -def post_add_sticker_into_sticker_pack( - request_data: RequestData, - settings: Settings, -) -> Response: - """Handle adding of sticker to sticker pack request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of adding. - """ - payload = add_sticker.AddSticker.parse_obj(request_data) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - sticker_from_sticker_pack = stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ) - - return PydanticResponse( - APIResponse[stickers.Sticker]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(create_sticker_pack.CreateStickerPack) -def post_create_sticker_pack(request: Request, settings: Settings) -> Response: - """Handle creating of sticker pack request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with result of creating. - """ - payload = create_sticker_pack.CreateStickerPack.parse_obj(request.params) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - sticker_from_sticker_pack = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(edit_sticker_pack.EditStickerPack) -def post_edit_sticker_pack(request_data: RequestData, settings: Settings) -> Response: - """Handle editing of sticker pack request. - - Arguments: - request_data: parsed json data from request. - settings: application settings with storage. - - Returns: - Response with result of editing. - """ - payload = edit_sticker_pack.EditStickerPack.parse_obj(request_data) - add_request_to_collection(settings, payload) - - inserted_at = dt.fromisoformat("2019-08-29T11:22:48.358586+00:00") - pack_stickers = [ - stickers.Sticker( - id=uuid.uuid4(), - emoji="🐢", - link="http://some_link.com", - inserted_at=inserted_at, - updated_at=inserted_at, - ), - ] - sticker_from_sticker_pack = stickers.StickerPack( - id=uuid.uuid4(), - name="Test sticker pack", - preview=None, - public=False, - stickers_order=None, - stickers=pack_stickers, - inserted_at=inserted_at, - updated_at=inserted_at, - deleted_at=None, - ) - - return PydanticResponse( - APIResponse[stickers.StickerPack]( - result=sticker_from_sticker_pack, - ), - ) - - -@bind_implementation_to_method(delete_sticker_pack.DeleteStickerPack) -def post_delete_sticker_pack(request: Request, settings: Settings) -> Response: - """Handle deleting of sticker pack request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with result of deleting. - """ - payload = delete_sticker_pack.DeleteStickerPack.parse_obj(request.params) - add_request_to_collection(settings, payload) - - return PydanticResponse( - APIResponse[str]( - result="sticker_pack_deleted", - ), - ) - - -@bind_implementation_to_method(delete_sticker.DeleteSticker) -def delete_sticker_from_sticker_pack(request: Request, settings: Settings) -> Response: - """Handle deleting of sticker from sticker pack request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with result of deleting. - """ - payload = delete_sticker.DeleteSticker.parse_obj(request.params) - add_request_to_collection(settings, payload) - - return PydanticResponse( - APIResponse[str]( - result="sticker_deleted", - ), - ) diff --git a/botx/testing/botx_mock/wsgi/routes/users.py b/botx/testing/botx_mock/wsgi/routes/users.py deleted file mode 100644 index 1aef2425..00000000 --- a/botx/testing/botx_mock/wsgi/routes/users.py +++ /dev/null @@ -1,71 +0,0 @@ -"""Endpoints for chats resource.""" - -from molten import Request, Response, Settings - -from botx.clients.methods.base import APIResponse -from botx.clients.methods.v3.users.by_email import ByEmail -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.clients.methods.v3.users.by_login import ByLogin -from botx.models.users import UserFromSearch -from botx.testing.botx_mock.binders import bind_implementation_to_method -from botx.testing.botx_mock.entities import create_test_user -from botx.testing.botx_mock.wsgi.messages import add_request_to_collection -from botx.testing.botx_mock.wsgi.responses import PydanticResponse - - -@bind_implementation_to_method(ByHUID) -def get_by_huid(request: Request, settings: Settings) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information about user. - """ - payload = ByHUID.parse_obj(dict(request.params)) - add_request_to_collection(settings, payload) - return PydanticResponse( - APIResponse[UserFromSearch]( - result=create_test_user(user_huid=payload.user_huid), - ), - ) - - -@bind_implementation_to_method(ByEmail) -def get_by_email(request: Request, settings: Settings) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information about user. - """ - payload = ByEmail.parse_obj(dict(request.params)) - add_request_to_collection(settings, payload) - return PydanticResponse( - APIResponse[UserFromSearch](result=create_test_user(email=payload.email)), - ) - - -@bind_implementation_to_method(ByLogin) -def get_by_login(request: Request, settings: Settings) -> Response: - """Handle retrieving information about user request. - - Arguments: - request: HTTP request from Molten. - settings: application settings with storage. - - Returns: - Response with information about user. - """ - payload = ByLogin.parse_obj(dict(request.params)) - add_request_to_collection(settings, payload) - return PydanticResponse( - APIResponse[UserFromSearch]( - result=create_test_user(ad=(payload.ad_login, payload.ad_domain)), - ), - ) diff --git a/botx/testing/building/__init__.py b/botx/testing/building/__init__.py deleted file mode 100644 index c32f8003..00000000 --- a/botx/testing/building/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Entities responsible for MessageBuilder.""" diff --git a/botx/testing/building/attachments.py b/botx/testing/building/attachments.py deleted file mode 100644 index 8b574ccb..00000000 --- a/botx/testing/building/attachments.py +++ /dev/null @@ -1,147 +0,0 @@ -"""Mixin for building attachments.""" -from dataclasses import field - -from botx.models import attachments as attach -from botx.testing import content as test_content - - -class BuildAttachmentsMixin: - """Mixin for building attachments in message.""" - - attachments: attach.AttachList = field(default_factory=list) # type: ignore - - def image( - self, - content: str = test_content.PNG_DATA, # noqa: WPS110 - file_name: str = "image.jpg", - ) -> None: - """Add image into incoming message. - - Arguments: - content: image content in RFC 2397 format. - file_name: name of file. - """ - self.attachments.__root__.append( - attach.ImageAttachment( - data=attach.Image(content=content, file_name=file_name), - ), - ) - - def document( - self, - content: str = test_content.DOCX_DATA, # noqa: WPS110 - file_name: str = "document.docx", - ) -> None: - """Add document into incoming message. - - Arguments: - content: document content in RFC 2397 format. - file_name: name of file. - """ - self.attachments.__root__.append( - attach.DocumentAttachment( - data=attach.Document(content=content, file_name=file_name), - ), - ) - - def location( - self, - location_name: str = "loc_name", - location_address: str = "loc_address", - location_lat: int = 0, - location_lng: int = 0, - ) -> None: - """Add location into incoming message. - - Arguments: - location_name: name of location. - location_lat: latitude. - location_lng: longitude. - location_address: address of location. - """ - self.attachments.__root__.append( - attach.LocationAttachment( - data=attach.Location( - location_name=location_name, - location_lat=location_lat, - location_lng=location_lng, - location_address=location_address, - ), - ), - ) - - def voice( - self, - content: str = test_content.MP3_DATA, # noqa: WPS110 - duration: int = 10, # noqa: WPS110 - ) -> None: - """Add voice into incoming message. - - Arguments: - content: voice content in RFC 2397 format. - duration: voice duration. - """ - self.attachments.__root__.append( - attach.VoiceAttachment( - data=attach.Voice(duration=duration, content=content), - ), - ) - - def video( - self, - content: str = test_content.MP4_DATA, # noqa: WPS110 - file_name: str = "video.mp4", - duration: int = 10, - ) -> None: - """Add video into incoming message. - - Arguments: - content: voice content in RFC 2397 format. - file_name: name of file. - duration: video duration. - """ - self.attachments.__root__.append( - attach.VideoAttachment( - data=attach.Video( - content=content, - file_name=file_name, - duration=duration, - ), - ), - ) - - def link( - self, - url: str = "https://google.com", - url_preview: str = "https://image.jpg", - url_text: str = "foo", - url_title: str = "bar", - ) -> None: - """Add link into incoming message. - - Arguments: - url: link on resource. - url_preview: link on url preview. - url_text: presented text on link. - url_title: title of link. - """ - self.attachments.__root__.append( - attach.LinkAttachment( - data=attach.Link( - url=url, - url_preview=url_preview, - url_text=url_text, - url_title=url_title, - ), - ), - ) - - def contact(self, contact_name: str = "Foo") -> None: - """Add link into incoming message. - - Arguments: - contact_name: name of contact - """ - self.attachments.__root__.append( - attach.ContactAttachment(data=attach.Contact(contact_name=contact_name)), - ) diff --git a/botx/testing/building/builder.py b/botx/testing/building/builder.py deleted file mode 100644 index 16300c28..00000000 --- a/botx/testing/building/builder.py +++ /dev/null @@ -1,99 +0,0 @@ -"""Builder for messages in tests.""" - -import uuid -from dataclasses import field -from typing import Any, Optional - -from pydantic import BaseConfig, validator -from pydantic.dataclasses import dataclass - -from botx.models.attachments import AttachList -from botx.models.entities import EntityList -from botx.models.enums import ChatTypes, ClientPlatformEnum, CommandTypes -from botx.models.messages.incoming_message import ( - Command, - DeviceMeta, - IncomingMessage, - Sender, -) -from botx.testing.building.attachments import BuildAttachmentsMixin -from botx.testing.building.entites import BuildEntityMixin -from botx.testing.building.validators import ( - convert_to_acceptable_file, - validate_body_corresponds_command, - validate_command_type_corresponds_command, -) - - -def _build_default_user() -> Sender: - return Sender( - user_huid=uuid.uuid4(), - group_chat_id=uuid.uuid4(), - chat_type=ChatTypes.chat, - ad_login="test_user", - ad_domain="example.com", - username="Test User", - is_admin=True, - is_creator=True, - host="cts.example.com", - manufacturer="Google", - device="Chrome 87.0", - device_software="macOS 10.15.7", - device_meta=DeviceMeta( - pushes=False, - timezone="Asia/Novosibirsk", - permissions={"microphone": True, "notifications": False}, - ), - platform=ClientPlatformEnum.web, - platform_package_id=None, - app_version="1.15.52", - locale="en", - ) - - -class BuilderConfig(BaseConfig): - """Config for builder dataclass.""" - - validate_assignment = True - - -@dataclass(config=BuilderConfig) -class MessageBuilder(BuildAttachmentsMixin, BuildEntityMixin): # noqa: WPS214 - """Builder for command message for bot.""" - - bot_id: uuid.UUID = field(default_factory=uuid.uuid4) - - command_data: dict = field(default_factory=dict) - system_command: bool = field(default=False) - file: Optional[Any] = field(default=None) - attachments: AttachList = field(default_factory=list) # type: ignore - user: Sender = field(default_factory=_build_default_user) - entities: EntityList = field(default_factory=list) # type: ignore - body: str = field(default="") - - _body_and_command_validator = validator("body", always=True)( - validate_body_corresponds_command, - ) - _command_type_and_data_validator = validator("system_command", always=True)( - validate_command_type_corresponds_command, - ) - _file_converter = validator("file", always=True)(convert_to_acceptable_file) - - @property - def message(self) -> IncomingMessage: - """Message that was built by builder.""" - command_type = CommandTypes.system if self.system_command else CommandTypes.user - command = Command( - body=self.body, - command_type=command_type, - data=self.command_data, - ) - return IncomingMessage( - sync_id=uuid.uuid4(), - command=command, - attachments=self.attachments, - file=self.file, - bot_id=self.bot_id, - user=self.user, - entities=self.entities, - ) diff --git a/botx/testing/building/entites.py b/botx/testing/building/entites.py deleted file mode 100644 index bf75b669..00000000 --- a/botx/testing/building/entites.py +++ /dev/null @@ -1,160 +0,0 @@ -"""Mixin for building entities.""" - -import uuid -from dataclasses import field -from datetime import datetime -from typing import Optional - -from botx.models.attachments_meta import DocumentAttachmentMeta -from botx.models.entities import ( - ChatMention, - Entity, - EntityList, - Forward, - Mention, - MentionTypes, - Reply, - UserMention, -) -from botx.models.enums import ChatTypes, EntityTypes -from botx.models.messages.message import Message - - -class BuildEntityMixin: - """Mixin for building entities in message.""" - - entities: EntityList = field(default_factory=list) # type: ignore - - def mention_contact(self, user_huid: uuid.UUID) -> None: - """Add contact mention to message for bot. - - Arguments: - user_huid: huid of user to mention. - """ - self.entities.__root__.append( - Entity( - type=EntityTypes.mention, - data=Mention( - mention_data=UserMention(user_huid=user_huid), - mention_type=MentionTypes.contact, - ), - ), - ) - - def mention_user(self, user_huid: uuid.UUID) -> None: - """Add user mention to message for bot. - - Arguments: - user_huid: huid of user to mention. - """ - self.entities.__root__.append( - Entity( - type=EntityTypes.mention, - data=Mention(mention_data=UserMention(user_huid=user_huid)), - ), - ) - - def mention_chat(self, chat_id: uuid.UUID) -> None: - """Add chat mention to message for bot. - - Arguments: - chat_id: id of chat to mention. - """ - self.entities.__root__.append( - Entity( - type=EntityTypes.mention, - data=Mention( - mention_data=ChatMention(group_chat_id=chat_id), - mention_type=MentionTypes.chat, - ), - ), - ) - - def mention(self, mention: Mention) -> None: - """Add mention by mention model. - - Arguments: - mention: mention model to build. - """ - self.entities.__root__.append(Entity(type=EntityTypes.mention, data=mention)) - - def reply( - self, - *, - message: Optional[Message] = None, - reply: Optional[Reply] = None, - source_chat_name: str = "chat", - ) -> None: - """Add reply to message for bot. - - Arguments: - message: replied message. - reply: reply model to build. - source_chat_name: name of chat where message was reply. - - Raises: - ValueError: raise if conflict of requirement arguments. - """ - if message and not reply: - mentions = message.entities.mentions - - reply = Reply( - attachment=DocumentAttachmentMeta(file_name="test.doc"), - body=message.body, - mentions=mentions, - reply_type=ChatTypes(message.chat_type), - sender=message.user_huid, # type: ignore - source_chat_name=source_chat_name, - source_sync_id=message.sync_id, - source_group_chat_id=message.group_chat_id, - ) - elif reply and not message: - pass # noqa: WPS420 - else: - raise ValueError("Must be replied message of reply model") - self.entities.__root__.append(Entity(type=EntityTypes.reply, data=reply)) - - def forward( - self, - *, - message: Optional[Message] = None, - forward: Optional[Forward] = None, - source_chat_name: str = "chat", - source_inserted_at: datetime = datetime( # noqa: B008, WPS404 - 1, - 1, - 1, - 1, - 1, - 1, - 1, - None, - ), - ) -> None: - """Add forward to message for bot. - - Arguments: - message: forwarded message. - forward: forward model to build. - source_chat_name: name of chat where message was forward. - source_inserted_at: ts of forwarded message. - - Raises: - ValueError: raise if conflict of requirement arguments. - """ - if message and not forward: - assert message.group_chat_id is not None - forward = Forward( - group_chat_id=message.group_chat_id, - sender_huid=message.user_huid or uuid.uuid4(), - forward_type=ChatTypes(message.chat_type), - source_chat_name=source_chat_name, - source_sync_id=message.sync_id, - source_inserted_at=source_inserted_at, - ) - elif forward and not message: - pass # noqa: WPS420 - else: - raise ValueError("Must be forwarding message or forward") - - self.entities.__root__.append(Entity(type=EntityTypes.forward, data=forward)) diff --git a/botx/testing/building/validators.py b/botx/testing/building/validators.py deleted file mode 100644 index 3d1ea8d7..00000000 --- a/botx/testing/building/validators.py +++ /dev/null @@ -1,119 +0,0 @@ -"""Validators and converters for fields in builder.""" - -from typing import Any, BinaryIO, Dict, Optional, TextIO, Union - -from botx.models import enums, events, files -from botx.models.messages.incoming_message import Sender - - -def validate_body_corresponds_command(body: str, values: dict) -> str: # noqa: WPS110 - """Check that passed body can be proceed. - - Arguments: - body: passed body. - values: already validated validated_values. - - Returns: - Checked passed body. - """ - _check_system_command_properties( - body, - values.get("system_command", False), - values["command_data"], - values, - ) - return body - - -def validate_command_type_corresponds_command( - is_system_command: bool, - values: dict, # noqa: WPS110 -) -> bool: - """Check that command type corresponds body. - - Arguments: - is_system_command: is command marked as system command. - values: already validated validated_values. - - Returns: - Checked flag. - """ - if is_system_command: - _check_system_command_properties( - values["body"], - is_system_command, - values["command_data"], - values, - ) - - return is_system_command - - -def convert_to_acceptable_file( - file: Optional[Union[files.File, BinaryIO, TextIO]], -) -> Optional[files.File]: - """Convert file to File that can be passed into message. - - Arguments: - file: passed file. - - Returns: - Converted file. - """ - if isinstance(file, files.File) or file is None: - return file - - new_file = files.File.from_file(file, filename="temp.txt") - new_file.file_name = file.name - return new_file - - -def _check_system_command_properties( - body: str, - is_system_command: bool, - command_data: dict, - validated_values: dict, -) -> None: - if is_system_command: - event = enums.SystemEvents(body) # check that is real system event - event_shape = events.EVENTS_SHAPE_MAP.get(event) - if event_shape is not None: - event_shape.parse_obj(command_data) # check event data - _event_checkers[event](**validated_values) # type: ignore - - -def _check_common_system_event(user: Sender, **_kwargs: Any) -> None: - error_field = "" - if user.user_huid is not None: - error_field = "user_huid" - elif user.ad_login is not None: - error_field = "ad_login" - elif user.ad_domain is not None: - error_field = "ad_domain" - elif user.username is not None: - error_field = "username" - - if error_field: - raise ValueError( - "user in system:chat_created can not have {0}".format(error_field), - ) - - -def _check_file_transfer_event(file: Optional[files.File], **_kwargs: Any) -> None: - if file is None: - raise ValueError("file_transfer event should have attached file") - - -def _check_internal_notification_event( - command_data: Dict[str, Any], - **_kwargs: Any, -) -> None: - assert "data" in command_data - - -_event_checkers = { - enums.SystemEvents.chat_created: _check_common_system_event, - enums.SystemEvents.added_to_chat: _check_common_system_event, - enums.SystemEvents.file_transfer: _check_file_transfer_event, - enums.SystemEvents.internal_bot_notification: _check_internal_notification_event, -} diff --git a/botx/testing/content.py b/botx/testing/content.py deleted file mode 100644 index eae291a9..00000000 --- a/botx/testing/content.py +++ /dev/null @@ -1,25 +0,0 @@ -"""Module with test data content for files.""" - -JPG_DATA = "" -JPEG_DATA = "" -GIF_DATA = "" -PNG_DATA = "" -DOC_DATA = "data:application/msword;base64,Lg==" -DOCX_DATA = "data:application/vnd.openxmlformats-officedocument.wordprocessingml.document;base64,Lg==" -XLS_DATA = "data:application/vnd.ms-excel;base64,Lg==" -XLSX_DATA = ( - "data:application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;base64,Lg==" -) -TXT_DATA = "data:text/plain;base64,Lg==" -PDF_DATA = "data:application/pdf;base64,Lg==" -HTML_DATA = "data:text/html;base64,Lg==" -JSON_DATA = "data:application/json;base64,Lg==" -SIG_DATA = "data:application/pgp-signature;base64,Lg==" -PPT_DATA = "data:application/vnd.ms-powerpoint;base64,Lg==" -PPTX_DATA = "data:application/vnd.openxmlformats-officedocument.presentationml.presentation;base64,Lg==" -MP3_DATA = "data:audio/mpeg;base64,Lg==" -MP4_DATA = "data:video/mp4;base64,Lg==" -GZ_DATA = "data:text/plain;base64,Lg==" -TGZ_DATA = "data:application/x-tar;base64,Lg==" -ZIP_DATA = "data:application/zip;base64,Lg==" -RAR_DATA = "data:application/vnd.rar;base64,Lg==" diff --git a/botx/testing/testing_client/__init__.py b/botx/testing/testing_client/__init__.py deleted file mode 100644 index 089d6c46..00000000 --- a/botx/testing/testing_client/__init__.py +++ /dev/null @@ -1 +0,0 @@ -"""Definition of client for testing.""" diff --git a/botx/testing/testing_client/base.py b/botx/testing/testing_client/base.py deleted file mode 100644 index 3b608876..00000000 --- a/botx/testing/testing_client/base.py +++ /dev/null @@ -1,125 +0,0 @@ -"""Base for testing client for bots.""" -from __future__ import annotations - -import asyncio -from concurrent.futures import ThreadPoolExecutor -from contextlib import contextmanager -from typing import Any, Dict, Generator, List, Optional, Tuple, Type - -import httpx - -from botx.bots.bots import Bot -from botx.clients.methods.base import BotXMethod -from botx.middlewares.exceptions import ExceptionMiddleware -from botx.models.messages.incoming_message import IncomingMessage -from botx.models.messages.message import Message -from botx.testing.botx_mock.asgi.application import get_botx_asgi_api -from botx.testing.botx_mock.wsgi.application import get_botx_wsgi_api -from botx.testing.typing import APIMessage, APIRequest - - -class _ExceptionMiddleware(ExceptionMiddleware): - """Replacement of built-in ExceptionMiddleware that will raise errors.""" - - async def _handle_error_in_handler(self, exc: Exception, message: Message) -> None: - handler = self._lookup_handler_for_exception(exc) - - if handler is None: - raise exc - - await super()._handle_error_in_handler(exc, message) - - -ErrorsOverrides = Dict[Type[BotXMethod], Tuple[int, Any]] - - -class BaseTestClient: - """Base for testing client for bots.""" - - def __init__( - self, - bot: Bot, - errors: Optional[ErrorsOverrides] = None, - suppress_errors: bool = False, - ) -> None: - """Init client with required query_params. - - Arguments: - bot: bot that should be tested. - errors: errors that should be raised from methods calls. - suppress_errors: if True then don't raise raise errors from handlers. - """ - self.bot: Bot = bot - self._original_http_client = bot.client.http_client - self._original_sync_http_client = bot.sync_client.http_client - self._error_middleware: Optional[ExceptionMiddleware] = None - self._messages: List[APIMessage] = [] - self._requests: List[APIRequest] = [] - self._errors = errors or {} - self._suppress_errors = suppress_errors - - def __enter__(self) -> BaseTestClient: - """Mock original HTTP clients.""" - is_error_middleware = isinstance( - self.bot.exception_middleware, - ExceptionMiddleware, - ) - if not self._suppress_errors and is_error_middleware: - self._error_middleware = self.bot.exception_middleware - self.bot.exception_middleware = _ExceptionMiddleware( - self.bot.exception_middleware.executor, - ) - self.bot.exception_middleware._exception_handlers = ( # noqa: WPS437 - self._error_middleware._exception_handlers # noqa: WPS437 - ) - - self.bot.client.http_client = httpx.AsyncClient( - app=get_botx_asgi_api(self._messages, self._requests, self._errors), - ) - self.bot.sync_client.http_client = httpx.Client( - app=get_botx_wsgi_api(self._messages, self._requests, self._errors), - ) - - return self - - def __exit__(self, *_: Any) -> None: - """Restore original HTTP client and clear storage.""" - if self._error_middleware is not None: - self.bot.exception_middleware = self._error_middleware - - ThreadPoolExecutor().submit( - asyncio.run, - self.bot.client.http_client.aclose(), - ).result() - self.bot.client.http_client = self._original_http_client - self.bot.sync_client.http_client = self._original_sync_http_client - self._messages = [] - - @contextmanager - def error_client( - self, - errors: Dict[Type[BotXMethod], Tuple[int, Any]], - ) -> Generator[BaseTestClient, None, None]: - """Enter into new test client that adds error responses to mocks. - - Arguments: - errors: overrides for errors in context. - - Yields: - New client with overridden errors. - """ - override_errors = {**self._errors, **errors} - with self.__class__(self.bot, override_errors, self._suppress_errors) as client: - yield client - - async def send_command(self, message: IncomingMessage, sync: bool = True) -> None: - """Send command message to bot. - - Arguments: - message: message with command for bot. - sync: if is `True` then wait while command is full executed. - """ - await self.bot.execute_command(message.dict()) - - if sync: - await self.bot.wait_current_handlers() diff --git a/botx/testing/testing_client/client.py b/botx/testing/testing_client/client.py deleted file mode 100644 index a1e18aa4..00000000 --- a/botx/testing/testing_client/client.py +++ /dev/null @@ -1,82 +0,0 @@ -"""Definition of client for testing.""" -from typing import Tuple, Union - -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.clients.methods.v3.events.edit_event import EditEvent -from botx.clients.methods.v3.events.reply_event import ReplyEvent -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.methods.v3.notification.notification import Notification -from botx.testing.testing_client.base import BaseTestClient -from botx.testing.typing import APIMessage, APIRequest - - -class TestClient(BaseTestClient): - """Test client for testing bots.""" - - # https://docs.pytest.org/en/latest/changelog.html#changes - # Allow to skip test classes from being collected - __test__: bool = False - - @property - def requests(self) -> Tuple[APIRequest, ...]: - """Return all requests that were sent by bot. - - Returns: - Sequence of requests that were sent from bot. - """ - return tuple(request.copy(deep=True) for request in self._requests) - - @property - def messages(self) -> Tuple[APIMessage, ...]: - """Return all entities that were sent by bot. - - Returns: - Sequence of messages that were sent from bot. - """ - return tuple(message.copy(deep=True) for message in self._messages) - - @property - def command_results(self) -> Tuple[CommandResult, ...]: - """Return all command results that were sent by bot. - - Returns: - Sequence of command results that were sent from bot. - """ - return tuple( - message for message in self.messages if isinstance(message, CommandResult) - ) - - @property - def notifications(self) -> Tuple[Union[Notification, NotificationDirect], ...]: - """Return all notifications that were sent by bot. - - Returns: - Sequence of notifications that were sent by bot. - """ - return tuple( - message - for message in self.messages - if isinstance(message, (Notification, NotificationDirect)) - ) - - @property - def message_updates(self) -> Tuple[EditEvent, ...]: - """Return all updates that were sent by bot. - - Returns: - Sequence of updates that were sent by bot. - """ - return tuple( - message for message in self.messages if isinstance(message, EditEvent) - ) - - @property - def replies(self) -> Tuple[ReplyEvent, ...]: - """Return all replies that were sent by bot. - - Returns: - Sequence of replies that were sent by bot. - """ - return tuple( - message for message in self.messages if isinstance(message, ReplyEvent) - ) diff --git a/botx/testing/typing.py b/botx/testing/typing.py deleted file mode 100644 index c7be802f..00000000 --- a/botx/testing/typing.py +++ /dev/null @@ -1,77 +0,0 @@ -"""Typings for test client and mocks.""" - -from typing import Union - -from botx.clients.methods.v2.bots import token -from botx.clients.methods.v3.chats import ( - add_admin_role, - add_user, - chat_list, - create, - info, - remove_user, - stealth_disable, - stealth_set, -) -from botx.clients.methods.v3.command import command_result -from botx.clients.methods.v3.events import edit_event, reply_event -from botx.clients.methods.v3.files import download, upload -from botx.clients.methods.v3.notification import direct_notification, notification -from botx.clients.methods.v3.stickers import ( - add_sticker, - create_sticker_pack, - delete_sticker, - delete_sticker_pack, - edit_sticker_pack, - sticker, - sticker_pack, - sticker_pack_list, -) -from botx.clients.methods.v3.users import by_email, by_huid, by_login - -APIMessage = Union[ - command_result.CommandResult, - notification.Notification, - direct_notification.NotificationDirect, - edit_event.EditEvent, - reply_event.ReplyEvent, -] - -APIRequest = Union[ - # V2 - # bots - token.Token, - # V3 - # chats - add_admin_role.AddAdminRole, - add_user.AddUser, - chat_list.ChatList, - info.Info, - remove_user.RemoveUser, - stealth_disable.StealthDisable, - stealth_set.StealthSet, - create.Create, - # command - command_result.CommandResult, - # notification - notification.Notification, - direct_notification.NotificationDirect, - # events - edit_event.EditEvent, - # users - by_huid.ByHUID, - by_email.ByEmail, - by_login.ByLogin, - # files - upload.UploadFile, - download.DownloadFile, - # stickers - sticker_pack_list.GetStickerPackList, - sticker_pack.GetStickerPack, - sticker.GetSticker, - add_sticker.AddSticker, - delete_sticker_pack.DeleteStickerPack, - delete_sticker.DeleteSticker, - create_sticker_pack.CreateStickerPack, - edit_sticker_pack.EditStickerPack, -] diff --git a/botx/typing.py b/botx/typing.py deleted file mode 100644 index dea6e83f..00000000 --- a/botx/typing.py +++ /dev/null @@ -1,35 +0,0 @@ -"""Aliases for complex types from `typing`.""" - -from typing import TYPE_CHECKING, Awaitable, Callable, TypeVar, Union - -from botx.models.messages import message - -if TYPE_CHECKING: - from botx.bots.bots import Bot # noqa: WPS433 - -try: - from typing import Literal # noqa: WPS433 -except ImportError: - from typing_extensions import Literal # type: ignore # noqa: WPS433, WPS440, F401 - -ExceptionT = TypeVar("ExceptionT", bound=Exception) - -# Something that can handle new message -AsyncExecutor = Callable[[message.Message], Awaitable[None]] -SyncExecutor = Callable[[message.Message], None] -Executor = Union[AsyncExecutor, SyncExecutor] - -# Middlware dispatchers -AsyncMiddlewareDispatcher = Callable[[message.Message, AsyncExecutor], Awaitable[None]] -SyncMiddlewareDispatcher = Callable[[message.Message, SyncExecutor], None] -MiddlewareDispatcher = Union[AsyncMiddlewareDispatcher, SyncMiddlewareDispatcher] - -# Exception handlers -AsyncExceptionHandler = Callable[[ExceptionT, message.Message], Awaitable[None]] -SyncExceptionHandler = Callable[[ExceptionT, message.Message], None] -ExceptionHandler = Union[AsyncExceptionHandler, SyncExceptionHandler] - -# Startup and shutdown events -AsyncLifespanEvent = Callable[["Bot"], Awaitable[None]] -SyncLifespanEvent = Callable[["Bot"], None] -BotLifespanEvent = Union[AsyncLifespanEvent, SyncLifespanEvent] diff --git a/docs/botx_api.md b/docs/botx_api.md new file mode 100644 index 00000000..29e40ec8 --- /dev/null +++ b/docs/botx_api.md @@ -0,0 +1,19 @@ +# BotX API + +--- + +## [Bots API](https://hackmd.ccsteam.ru/s/E9MPeOxjP#Bots-API) + +### [Получение токена](https://hackmd.ccsteam.ru/s/E9MPeOxjP#%D0%9F%D0%BE%D0%BB%D1%83%D1%87%D0%B5%D0%BD%D0%B8%D0%B5-%D1%82%D0%BE%D0%BA%D0%B5%D0%BD%D0%B0) + +Токен можно получить для каждого из добавленных аккаунтов бота. Для выбора аккаунта +используется его ID. + +!!! note + + Вряд ли вам когда-нибудь понадобится запрашивать токен вручную, `pybotx` + получает их автоматически. + +``` py +--8<-- "docs/snippets/client/bots_api/get_token.py" +``` diff --git a/docs/changelog.md b/docs/changelog.md deleted file mode 100644 index d207ca3f..00000000 --- a/docs/changelog.md +++ /dev/null @@ -1,864 +0,0 @@ -## 0.28.0 (Nov 11, 2021) - -### Added - -* SmartApps main functionality. - - -## 0.27.0 (Nov 8, 2021) - -### Added - -* `pin_message` and `unpin_message` methods. - - -## 0.26.0 (Nov 1, 2021) - -### Added - -* Added methods for interacting with sticker pack/stickers - `get_sticker_pack_list`, `get_sticker_pack`. `get_sticker_from_pack`, `create_sticker_pack`, `add_sticker`, `edit_sticker_pack`, `delete_sticker_pack`, `delete_sticker`. - - -## 0.25.1 (Oct 22, 2021) - -### Changed - -* Add `embed_mentions` argument in `answer_message` method. - - -## 0.25.0 (Sep 17, 2021) - -### Added - -* `cts_login` and `cts_logout` system events. - - -## 0.24.0 (Sep 14, 2021) - -### Removed - -* File extensions validation. -* `File.has_supported_extension` classmethod. - -### Added - -* Multiple mime-types. - -### Changed - -* `File.get_ext_by_mimetype` now don't raise `ValueError` and returns `None` if mimetype not found. - - -## 0.23.2 (Sep 09, 2021) - -### Added - -* Add `file_name` param to `download_file` method to provide ability to change the returned file name. - - -## 0.23.1 (Aug 30, 2021) - -### Fixed - -* Memory leak in bot.tasks collection. - - -## 0.23.0 (Aug 23, 2021) - -### Added - -* Add method for uploading files to chat. -* Add method for downloading files from chat. - -### Changed - -* Add `data` and `files` fields to `HTTPRequest` for sending multipart/form-data in request. -* Add `expected_type` field to `HTTPRequest` and `expected_type` property to `BaseBotXMethod` - to allow interacting with non JSON responses. -* Add `should_process_as_error` field to `HTTPRequest` so that errors that are not in - the range of 400 to 599 can be added. -* Add `raw_data` to `HTTPResponse`so that you can process raw content of response. - - -## 0.22.1 (Aug 23, 2021) - -### Fixed - -* Add `embed_mentions` argument in `SendingMessage.from_message` method. -* Fix `EMBED_MENTION_RE` expression. - - -## 0.22.0 (Aug 19, 2021) - -Tested on BotX 1.44.0-rc2 - -### Added - -* Sending and handling internal bot notifications. - - -## 0.21.3 (Aug 17, 2021) - -### Fixed - -* Bot's method `authorize()` now not fall if cant take some tokens. Just logging and skip invalid credentials. - - -## 0.21.2 (Aug 3, 2021) - -### Fixed - -* `File` is now serializing when sending message. - - -## 0.21.1 (Jul 28, 2021) - -### Fixed - -* Make the `body` attribute of `Reply` event optional. -* Add `AttachmentMeta` model to `Reply` event instead of `Attachments`. - - -## 0.21.0 (Jul 23, 2021) - -Tested on BotX 1.42.0-rc4 - -### Fixed - -* Remove `Dict[str, Any]` from type of `error_data` field of `BotDisabledResponse`, - now it can only be `BotDisabledErrorData`. - - -## 0.20.4 (Jul 23, 2021) - -Tested on BotX 1.42.0-rc4 - -### Added - -* Add possibility to send message that visible only in stealth mode. -* Add support for embed mentions (can be used anywhere in text). - -### Fixed - -* Fix silent response by changing option location in method call. - - -## 0.20.3 (Jul 22, 2021) - -Tested on BotX 1.42.0-rc4 - -### Add - -* Add possibility to create chats with enabled shared_history option (Bot.create_chat). - - -## 0.20.2 (Jul 22, 2021) - -Tested on BotX 1.42.0-rc4 - -### Changed - -* Exceptions thrown in `exception_handler` are now logged. - - -## 0.20.1 (Jul 19, 2021) - -Tested on BotX 1.42.0-rc4 - -### Added - -* Add `bot not found` error handler for `Token` method. -* Add `invalid bot credentials` error handler for `Token` method. -* Add `connection error` handler for all BotX methods. -* Add `JSON decoding error` handler for all BotX methods. - - -## 0.20.0 (Jul 08, 2021) - -Tested on BotX 1.42.0-rc4 - -### Added - -* Add method for retrieving list of bot's chats -* Add `inserted_at` field to `ChatFromSearch` model - -### Changed - -* `HTTPRequest` & `HTTPResponse` moved to `clients.types` -* `HTTPRequest` now work with JSON (dict) instead of bytes. It improves consistency with - `HTTPResponse` and will be useful in interceptors implementation. -* Reply event field `source_chat_name` is optional now -* Forward event field `source_sync_id` is required now - -### Removed - -* `HTTPResponse` bytes content property - -### Fixed - -* `File` is now deleted when the message is updated -* `Bot_id` is now displayed in the request -* Add description for `BotXAPIError` - - -## 0.19.1 (May 21, 2021) - -Tested on BotX 1.40.0-rc0 - -### Changed - -* `method.host` assignment moved to method constructor in unit-tests - - -## 0.19.0 (May 18, 2021) - -Tested on BotX 1.40.0-rc0 - -### Added - -* Bot now has method `authorize` for explicit authorization each account -* Add `source_sync_id` field to `Message` which contains id of the message if it was sent from button - -### Changed - -* `ExpressServer` renamed to `BotXCredentials` -* Bot id now required for `BotXCredentials` - -### Removed - -* `send_from_button` field from `Message` -* `ui` flag from button's data - -### Fixed - -* Fix pydantic deprecation warning: `whole` flag renamed to `each_item` - -## 0.18.4 (Apr 08, 2021) - -Tested on BotX 1.40.0-rc0 - -### Changed - -* Raise special exception (BotXAPIRouteDeprecated) on response 410 GONE - -## 0.18.3 (Apr 05, 2021) - -### Added - -* Edit events and answer message now support metadata - -## 0.18.2 (Apr 01, 2021) - -### Changed - -* Bot can recognize mention all (@all) now - -## 0.18.1 (Mar 22, 2021) - -### Changed - -* Fixed empty label bug, now you can use empty string as button label - -## 0.18.0 (Mar 13, 2021) - -### Changed - -* Now `message.data` returns concatenated metadata and data from UI element -* `message.user.email` renamed to `message.user.upn` - -## 0.17.1 (Feb 9, 2021) - -### Added - -* Add `handler` option for bubbles and keyboard buttons - - -## 0.17.0 (Jan 28, 2021) - -### Changed - -* Mimetypes are now taken from constant dict, not `mimetypes` library -* All models use enum's fields by value - -### Fixed - -* `File.media_type` property - -### Added - -* Now `File` has property `size_in_bytes` -* New fields in **message.user**: `manufacturer`, `device`, `device_software`, - `device_meta`, `platform`, `platform_package_id`, `app_version`, `locale`. - - -## 0.16.10 (Dec 29, 2020) - -### Added - -* Attachments now have property `attach_type` with type of attachment - - -## 0.16.9 (Dec 29, 2020) - -### Fixed - -* Dependencies added trought `include_collector` don't called - - -## 0.16.8 (Dec 24, 2020) - -### Added - -* `async_from_file`, `file_chunks` methods for `File` to async work with attachments -* new dependency "base64io" - -### Changed - -* `from_file` now uses stream base64 encoding - - -## 0.16.7 (Dec 23, 2020) - -### Fixed - -* has_supported_extension now supports uppercase extensions - - -## 0.16.6 (Dec 22, 2020) - -### Changed - -* Attachments now have default value for type - -### Fixed - -* Default handler don't process system events anymore - - -## 0.16.5 (Dec 22, 2020) - -### Added - -* New static method to check that file extension can be handled by BotX API - - -## 0.16.4 (Dec 16, 2020) - -### Fixed - -* Fixed casting Voice attachment to Video - - -## 0.16.3 (Dec 14, 2020) - -### Fixed - -* Now you can accept File with unsupported extension - - -## 0.16.2 (Dec 11, 2020) - -### Fixed - -* Now you can reply with mention (no text or file) - - -## 0.16.1 (Dec 09, 2020) - -### Added - -* StatusRecipient import from core module - - -## 0.16.0 (Dec 07, 2020) - -### Added - -* Support of attachments in messages for bot's api v4 -* Support of reply in messages for bot's api v4 -* Builder of attachments in MessageBuilder -* Test content in RFC 2397 format -* Entity building methods for `MessageBuilder` -* Flag `is_forward` for `Message` -* Bot's method `reply` for reply by message - -### Changed - -* Type of `message.entities` from `List[Attachment]` to is `EntityList` -* `send()` sending only direct notification, not command result - - -## 0.15.17 (Nov 20, 2020) - -### Changed - -* Now you can update attached file while updating message - - -## 0.15.16 (Nov 19, 2020) - -### Added - -* New event `left_from_chat`, which stores HUIDs of members who left the chat - - -## 0.15.15 (Nov 17, 2020) - -### Added - -* New event `deleted_from_chat`, which stores deleted members HUIDs - - -## 0.15.14 (Nov 13, 2020) - -### Fixed - -* `added_to_chat` event is created properly now - - -## 0.15.13 (Nov 6, 2020) - -## Added - -* Add `bot_id` to outgoing messages debug logs. Incoming messages already have it - - -## 0.15.12 (Oct 30, 2020) - -### Added - -* Added width weight for bubbles and keyboard buttons (`h_size`). The more width weight, - the more horizontal space is occupied by button. - - -## 0.15.11 (Oct 30, 2020) - -### Added - -* Added `.sig`, `.mp3` and `.mp4` to allowed file formats - - -## 0.15.10 (Oct 29, 2020) - -### Added - -* Added alerts (toast on button press) for bubbles and keyboard buttons - - -## 0.15.9 (Oct 28, 2020) - -### Added - -* New message option 'silent_response' to hide next user messages in chat - - -## 0.15.8 (Oct 23, 2020) - -### Changed - -* `httpx` dependency was updated to 0.16 -* `loguru` dependency was updated to 0.5 - - -## 0.15.7 (Oct 22, 2020) - -### Added - -* New bot method `add_admin_roles` for promoting users to admins (bot should have - admin role itself) - - -## 0.15.6 (Oct 12, 2020) - -### Fixed - -* Fix `system:chat_created` event (`UserInChatCreated.name` now optional) - - -## 0.15.5 (Sep 16, 2020) - -### Added - -* Added `.ppt` and `.pptx` to allowed file formats - - -## 0.15.4 (Sep 4, 2020) - -### Added - -* Update accepted extensions list (`.jpg`, `.jpeg`, `.gif`, `.png`, `.json`, `.zip`, `.rar`) - - -## 0.15.3 (Aug 26, 2020) - -### Added - -* Models were added to API Reference - -### Fixed - -* Fix "handler not found" error message - - -## 0.15.2 (Aug 7, 2020) - -### Fixed - -* Fix overwriting SendingMessage credentials - - -## 0.15.1 (Aug 4, 2020) - -### Fixed - -* Fix optional fields in UserFromSearch model - - -## 0.15.0 (Jul 23, 2020) - -### Added - -* Added startup and shutdown lifespan events. -* Added support for synchronous requests (now `molten` is required for tests). -* Added support for `system:added_to_chat` event. -* Added support for forwarded messages. -* Added support for passing additional arguments to `Bot.status()`. -* Added methods for: - * chat creation; - * information about chat retrieving; - * search user by email, user HUID or AD login/domain. -* Add client flag for logger. -* Allow to update message through `.send()` from bot. -* Add `metadata` property to message. - -### Fixed - -* Fix information about sender for `system:chat_created` event. -* Fix crash when forwarding message to bot. -* Fix error on creating empty credentials. - -### Changed - -* Simplify middlewares. Now sync middlewares will receive sync `call_next` and -asynchronous async `call_next`. -* `TestClient` will now propagate unhandled errors. -* Rewrite inner clients. They now work with `methods` classes. -* Update `httpx` to `^0.13.0` -* Use bot as default dependency overrides provider. -* Simplify cache key. - - -## 0.14.1 (Apr 29, 2020) - -### Added - -* Add flag for messages sent from buttons. - -### Fixed - -* Attach dependencies on default handler when collector is included by another. -* Suppress `KeyError` when dropping key from next steps storage. - -## 0.14.0 (Apr 3, 2020) - -### Added - -* Support for external entities in incoming message, like mentions. -* Add ability to specify custom message id for new message from bot. - -### Changed - -* Refactor exceptions to be more usefule. Now exceptions have additional properties with extra data. -* If there will be an error when convering function to handler or dependency, then exception will contain information about failed attributes. -* Collector will iterate through handlers in right order. -* Change deploing documentation to github pages from master branch. - -### Fixed - -* Fix shape for `bot_disabled` response. - -## 0.13.6 (Mar 20, 2020) - -### Added - -* Add Netlify for documention preview. -* Parse handler docstring as `full_description` for handler. -* Preserve order of added handlers. - -### Changed - -* Skip validation for incoming file, so files with unsupported extensions. - -### Fixed - -* Fix logging file. - -## 0.13.5 (Mar 6, 2020) - -### Added - -* Add channel type into `MentionTypes`. - -### Changed - -* Replace travis with github actions. - -### Fixed - -* Fix dependencies extending when use `Collector.include_collector` and dependencies are -defined in collector initialization. -* Fix default handler including into another collector. -* Fix message text logging. -* Fix internal links in docs. - -## 0.13.4 (Mar 3, 2020) - -### Added - -* Examples of bots that are built using `pybotx`: - * Bot that defines finite-state machine behaviour for handlers. - -### Changed - -* Log exception traceback with `logger.exception` instead of `logger.error` when error was -not caught. -* Default handler will be excluded from status by default (as it was in library versions before 0.13.0). - -## 0.13.3 (Feb 26, 2020) - -### Added - -* Add background dependencies to next step middleware. -* Next step break handler can be registered as function. -* Add methods to add/remove users to/from chat using `Bot.add_users_into_chat()` and `Bot.remove_users_from_chat()`. - -### Fixed - -* Add missing `dependency_overrides_provider` to `botx.collecting.Collector.add_handler`. -* Encode message update payload by alias. - -### Changed - -* Refactored next step middleware -* Next step middleware won't now lookup for handler in bot. -* Disable `loguru` logger by default. - -## 0.13.2 (Feb 14, 2020) - -### Fixed - -* Check that there are futures while stopping bot. -* Strip command in `Handler.command_for` so no space at the end. - -## 0.13.1 (Feb 6, 2020) - -### Added - -* Stealth mode enable/disable methods `Bot.enable_stealth_mode()` and `Bot.disable_stealth_mode()`. - -## 0.13.0 (Jan 20, 2020) - -!!! warning - A lot of breaking changes. See API reference for more information. - -### Added -* Added `Silent buttons`. -* Added `botx.TestClient` for writing tests for bots. -* Added `botx.MessageBuilder` for building message for tests. -* `botx.Bot` can now accept sequence of `collecting.Handler`. -* `botx.Message` is now not a pydantic model. Use `botx.IncomingMessage` in webhooks. - -### Changed - -* `AsyncBot` renamed to `Bot` in `botx.bots`. -* `.stop` method was renamed to `.shutdown` in `botx.Bot`. -* `UserKindEnum` renamed to `UserKinds`. -* `ChatTypeEnum` renamed to `ChatTypes`. - -### Removed - -* Removed `botx.bots.BaseBot`, only `botx.bots.Bot` is now available. -* Removed `botx.BotCredentials`. Credentials for bot should be registered via -sequence of `botx.ExpressServer` instances. -* `.credentials` property, `.add_credentials`, `.add_cts` methods were removed in `botx.Bot`. -Known hosts can be obtained via `.known_hosts` field. -* `.start` method in `botx.Bot` was removed. - -## 0.12.4 (Oct 12, 2019) - -### Added - -* Add `cts_user` value to `UserKindEnum`. - -## 0.12.3 (Oct 12, 2019) - -### Changed - -* Update `httpx` to `0.7.5`. -* Use `https` for connecting to BotX API. - -### Fixed - -* Remove reference about `HandlersCollector.regex_handler` from docs. - -## 0.12.2 (Sep 8, 2019) - -### Fixed - -* Clear `AsyncBot` tasks on shutdown. - -## 0.12.1 (Sep 2, 2019) - -### Changed - -* Upgrade `pydantic` to `0.32.2`. - -### Added - -* Added `channel` type to `ChatTypeEnum`. - -### Fixed - -* Export `UserKindEnum` from `botx`. - - -## 0.12.0 (Aug 30, 2019) - -### Changed - -* `HandlersCollector.system_command_handler` now takes an `event` argument of type `SystemEventsEnum` instead of the deleted argument `comamnd`. -* `MessageCommand.data` field will now automatically converted to events data types corresponding to special events, -such as creating a new chat with a bot. -* Replaced `requests` and `aiohttp` with `httpx`. -* Moved synchronous `Bot` to `botx.sync` module. The current `Bot` is an alias to the `AsyncBot`. -* `Bot.status` again became a coroutine to add the ability to receive different commands for different users -depending on different conditions defined in the handlers (to be added to future releases, when BotX API support comes up). -* Changed methods signatures. See `api-reference` for details. - -### Added - -* Added logging via `loguru`. -* `Bot` can now accept both coroutines and normal functions. -* Added mechanism for catching exceptions. -* Add ability to use sync and async functions to send data from `Bot`. -* Added dependency injection system -* Added parsing command query_params into handler arguments. - -### Removed - -* `system_command_handler` argument has been removed from the `HandlersCollector.handler` method. -* Dropped `aiojobs`. - -### Fixed - -* Fixed `opts` shape. - -## 0.11.3 (Jul 24, 2019) - -### Fixed - -* Catch `IndexError` when trying to get next step handler for the message and there isn't available. - -## 0.11.2 (Jul 17, 2019) - -### Removed - -* `.data` field in `BubbleElement` and `KeyboardElement` was removed to fix problem in displaying markup on some clients. - -## 0.11.1 (Jun 28, 2019) - -### Fixed - -* Exception won't be raised on successful status codes from the BotX API. - -## 0.11.0 (Jun 27, 2019) - -### Changed - -* `MkDocs` documentation and move to `github`. -* `BotXException` will be raised if there is an error in sending message, obtaining tokens, parsing incoming message data and some other cases. -* Rename `CommandRouter` to `HandlersCollector`, changed methods, added some new decorators for specific commands. -* Replaced `Bot.parse_status` method with the `Bot.status` property. -* Added generating message for `BotXException` error. - -### Added - -* `ReplyMessage` class and `.reply` method to bots were added for building answers in command in more comfortable way. -* Options for message notifications. -* Bot's handlers can be registered as next step handlers. -* `MessageUser` has now `email`. - -## 0.10.3 (May 31, 2019) - -### Fixed - -* Fixed passing positional and key arguments into logging wrapper for next step handlers. - -## 0.10.2 (May 31, 2019) - -### Added - -* Next step handlers can now receive positional and key arguments that are passed through their registration. - -## 0.10.1 (May 31, 2019) - -### Fixed - -* Return handler function from `CommandRouter.command` decorator instead of `CommandHandler` instance. - -## 0.10.0 (May 28, 2019) - -### Changed - -* Move `requests`, `aiohttp` and `aiojobs` to optional dependencies. -* All handlers now receive a bot instance that processes current command execution as second argument for handler. -* Files renamed using snake case. -* Returned response text and status from methods for sending messages. - -### Added - -* Export `pydantic`'s `ValidationError` directly from `botx`. -* Add Readme.md for library. -* Add support for BotX API tokens for bots. -* Add `py.typed` file for `mypy`. -* Add `CommandRouter` for gathering command handlers together and some methods for handling specific commands. -* Add ability to change handlers processing behaviour by using next step handlers. -* Add `botx.bots.Bot.answer_message` method to bots for easier generating answers in commands. -* Add mentions for users in chats. -* Add abstract methods to `BaseBot` and `BaseDispatcher`. - -### Fixed - -* Fixed some mypy types issues. -* Removed print. - -## 0.9.4 (Apr 23, 2019) - -### Changed - -* Change generation of command bodies for bot status by not forcing leading slash. - -## 0.9.3 (Apr 4, 2019) - -### Fixed - -* Close `aiohttp.client.ClientSession` when calling `AsyncBot.stop()`. - -## 0.9.2 (Mar 27, 2019) - -### Removed - -* Delete unused for now argument `bot` from thread wrapper. - -## 0.9.1 (Mar 27, 2019) - -### Fixed - -* Log unhandled exception from synchronous handlers. - -## 0.9.0 (Mar 18, 2019) - -### Added - -* First public release in PyPI. -* Synchronous and asynchronous API for building bots. diff --git a/docs/css/custom.css b/docs/css/custom.css new file mode 100644 index 00000000..f0c708c1 --- /dev/null +++ b/docs/css/custom.css @@ -0,0 +1,10 @@ +div.autodoc-docstring { + padding-left: 20px; + margin-bottom: 30px; + border-left: 5px solid rgba(230, 230, 230); +} + +div.autodoc-members { + padding-left: 20px; + margin-bottom: 15px; +} diff --git a/docs/development/collector.md b/docs/development/collector.md deleted file mode 100644 index f984b9d4..00000000 --- a/docs/development/collector.md +++ /dev/null @@ -1,55 +0,0 @@ -At some point you may decide that it is time to split your handlers into several files. -In order to make it as convenient as possible, `pybotx` provides a special mechanism that is similar to the mechanism -of routers from traditional web frameworks like `Blueprint`s in `Flask`. - -Let's say you have a bot in the `bot.py` file that has many commands (public, hidden, next step) which can be divided into 3 groups: - - * commands to access `A` service. - * commands to access `B` service. - * general commands for handling files, saving user settings, etc. - -Let's divide these commands in a following way: - - 1. Leave the general commands in the `bot.py` file. - 2. Move the commands related to `A` service to the `a_commands.py` file. - 3. Move commands related to `B` service to the `b_commands.py` file. - -### Collector - -[Collector][botx.collecting.collectors.collector.Collector] is a class that can collect registered handlers -inside itself and then transfer them to bot. - -Using [Collector][botx.collecting.collectors.collector.Collector] is quite simple: - - 1. Create an instance of the collector. - 2. Register your handlers, just like you do it for your bot. - 3. Include registered handlers in your [Bot][botx.bots.bots.Bot] instance using the [`.include_collector`][botx.bots.mixins.collectors.BotCollectingMixin.include_collector] method. - -Here is an example. - -If we have already divided our handlers into files, it will look something like this for the `a_commands.py` file: - -```Python3 -{!./src/development/collector/collector0/a_commands.py!} -``` - -And here is the `bot.py` file: - -```Python3 -{!./src/development/collector/collector0/bot.py!} -``` - -!!! warning - - If you try to add 2 handlers for the same command, `pybotx` will raise an exception indicating about merge error. - -### Advanced handlers registration - -There are different methods for handlers registration available on [Collector][botx.collecting.collectors.collector.Collector] and [Bot][botx.bots.bots.Bot] instances. -You can register: - -* regular handlers using [`.handler`][botx.collecting.collectors.mixins.handler.HandlerMixin.handler] decorator. -* default handlers, that will be used if matching handler was not found using [`.default`][botx.collecting.collectors.mixins.default.DefaultHandlerMixin.default] decorator. -* hidden handlers, that won't be showed in bot's menu using [`.hidden`][botx.collecting.collectors.mixins.hidden.HiddenHandlerMixin.hidden] decorator. -* system event handlers, that will be used for handling special events from BotX API using [`.system_event`][botx.collecting.collectors.mixins.system_events.SystemEventsHandlerMixin.system_event] decorator. -* and some other type of handlers. See API reference for bot or collector for more information. diff --git a/docs/development/dependencies-injection.md b/docs/development/dependencies-injection.md deleted file mode 100644 index a51592b4..00000000 --- a/docs/development/dependencies-injection.md +++ /dev/null @@ -1,35 +0,0 @@ -`pybotx` has a dependency injection mechanism heavily inspired by [`FastAPI`](https://fastapi.tiangolo.com/tutorial/dependencies/). - -## Usage - -First, create a function that will execute some logic. It can be a coroutine or a simple function. -Then write a handler for bot that will use this dependency: - -```python3 -{!./src/development/dependencies_injection/dependencies_injection0.py!} -``` - -## Dependencies with dependencies - -Each of your dependencies function can contain parameters with other dependencies. And all this will be solved at the runtime: - -```python3 -{!./src/development/dependencies_injection/dependencies_injection1.py!} -``` - -## Special dependencies: Bot and Message - -[Bot][botx.bots.bots.Bot] and `Message` objects and special case of dependencies. -If you put an annotation for them into your function then this objects will be passed inside. -It can be useful if you write something like authentication dependency: - -```python3 -{!./src/development/dependencies_injection/dependencies_injection2.py!} -``` - -[DependencyFailure][botx.exceptions.DependencyFailure] exception is used for preventing execution -of dependencies after one that failed. - -Also, if you define a list of dependencies objects in the initialization of [collector][botx.collecting.collectors.collector.Collector] or [bot][botx.bots.bots.Bot] or in `.handler` decorator or others, -then these dependencies will be processed as background dependencies. -They will be executed before the handler and its' dependencies: \ No newline at end of file diff --git a/docs/development/first-steps.md b/docs/development/first-steps.md deleted file mode 100644 index 41f520b6..00000000 --- a/docs/development/first-steps.md +++ /dev/null @@ -1,176 +0,0 @@ -Let's create a new bot, which will ask the user for his data, and then send some statistics from the collected information. -Take echo-bot, from the [Introduction](../index.md), and gradually improve it step by step. - -## Starting point - -Right now we have the following code: - -```Python3 -{!./src/development/first_steps/first_steps0.py!} -``` - -## First, let's see how this code works -We will explain only those parts that relate to `pybotx`, and not to the frameworks used in this documentation. - -### Step 1: import `Bot`, `Message`, `Status` and other classes - -```Python3 hl_lines="1" -{!./src/development/first_steps/first_steps0.py!} -``` - -* `Bot` is a class that provides all the core functionality to your bots. -* `Message` provides data to your handlers for commands. -* `Status` is used here only to document the `FastAPI` route, -but in fact it stores information about public commands that user of your bot should see in menu. -* `ExpressServer` is used for storing information -about servers with which your bot is able to communicate. -* `IncomingMessage` is a pydantic model that is used -for base validating of data, that was received on your bot's webhook. - -### Step 2: initialize your `Bot` - -```Python3 hl_lines="5" -{!./src/development/first_steps/first_steps0.py!} -``` - -The `bot` variable will be an "instance" of the class `Bot`. -We also register an instance of the cts server to get tokens and the ability to send requests to the API. - -### Step 3: define default handler - -```Python3 hl_lines="8" -{!./src/development/first_steps/first_steps0.py!} -``` - -This handler will be called for all commands that have not appropriate handlers. -We also set `include_in_status=False` so that handler won't be visible in menu and it won't -complain about "wrong" body generated for it automatically. - -### Step 4: send text to user - -```Python3 hl_lines="10" -{!./src/development/first_steps/first_steps0.py!} -``` - -[`.answer_message`][botx.bots.mixins.sending.SendingMixin.answer_message] will send text to the user by using -`sync_id`, `bot_id` and `host` data from the `Message` instance. -This is a simple wrapper for the [`.send`][botx.bots.mixins.sending.SendingMixin.send] method, which is used to -gain more control over sending messages process, allowing you to specify a different -host, bot_id, sync_id, group_chat_id or a list of them. - -### Step 5: register handler for bot proper shutdown. - -```Python3 hl_lines="14" -{!./src/development/first_steps/first_steps0.py!} -``` - -The `.shutdown` method is used to stop pending handler. -You must call them to be sure that the bot will work properly. - -### Step 6: define webhooks for bot - -```Python3 hl_lines="17 22" -{!./src/development/first_steps/first_steps0.py!} -``` - -Here we define 2 `FastAPI` routes: - - * `GET` on `/status` will tell BotX API which commands are available for your bot. - * `POST` on `/command` will receive data for incoming messages for your bot and execute handlers for commands. - -!!! info - - If `.execute_command` did not find a handler for - the command in the message, it will raise an `NoMatch` error in background, - which you probably want to [handle](./handling-errors.md). You can register default handler to process all commands that do not have their own handler. - -### Step 7 (Improvement): Reply to user if message was received from host, which is not registered - -We can send to BotX API a special response, that will say to user that bot can not communicate with -user properly, since message was received from unknown host. We do it by handling -[ServerUnknownError][botx.exceptions.ServerUnknownError] and returning to BotX API information -about error. - -```Python3 hl_lines="35" -{!./src/development/first_steps/first_steps1.py!} -``` - -## Define new handlers - -Let's define a new handler that will trigger a chain of questions for the user to collect information. - -We'll use the `/fill-info` command to start the chain: - -```Python3 hl_lines="14" -{!./src/development/first_steps/first_steps2.py!} -``` - -Here we define a new handler for `/fill-info` command using `.handler` decorator. -This decorator will generate for us body for our command and register it doing it available to handle. -We also defined a `users_data` dictionary to store information from our users. - -Now let's define another 2 handlers for the commands that were mentioned in the message text that we send to the user: - - * `/my-info` will just send the information that users have filled out about themselves. - * `/info` will send back the number of users who filled in information about themselves, their average age and number of male and female users. - * `/infomation` is an alias to `/info` command. - -```Python3 hl_lines="32 50" -{!./src/development/first_steps/first_steps3.py!} -``` - -Take a look at highlighted lines. `.handler` method takes a -different number of arguments. The most commonly used arguments are `command` and `commands`. -`command` is a single string that defines a command for a handler. -`commands` is a list of strings that can be used to define a variety of aliases for a handler. -You can use them together. In this case, they simply merge into one array as if you specified only `commands` argument. - -See also at how the commands themselves are declared: - - * for the `fill_info` function we have not defined any `command` but it will be implicitly converted to the `/fill-info` command. - * for the `get_info_for_user` function we had explicitly specified `/my-info` string. - * for the `get_processed_information` we specified a `commands` argument to define many aliases for the handler. - -## Register next step handlers - -`pybotx` provide you the ability to change mechanism of handlers processing by mechanism of -middlewares. It also provides a middleware for handling chains of messages by [`Next Step Middleware`][botx.middlewares.ns.NextStepMiddleware]. - -To use it you should define functions that will be used when messages that start chain will be handled. -All functions should be defined before bot starts to handler messages, since dynamic registration -can cause different hard to find problems. - -Lets' define these handlers and, finally, create a chain of questions from the bot to the user. - -First we should import our middleware and functions that will register function fo next -message from user. - -```Python3 hl_lines="2" -{!./src/development/first_steps/first_steps4.py!} -``` - -Next we should define our functions and register it in our middleware. - -```Python3 hl_lines="11 17 36 51 52 53" -{!./src/development/first_steps/first_steps4.py!} -``` - -And the last part of this step is use -[register_next_step_handler][botx.middlewares.ns.register_next_step_handler] function to -register handler for next message from user. - -```Python3 hl_lines="14 24 28 33 48 68" -{!./src/development/first_steps/first_steps4.py!} -``` - -### Recap - -What's going on here? We added one line to our `/fill-info` command to start a chain of -questions for our user. We also defined 3 functions, whose signature is similar to the -usual handler signature, but instead of registration them using the -`.handler` decorator, we do this while registering out -[`Next Step Middleware`][botx.middlewares.ns.NextStepMiddleware] for bot. We change message -handling flow using the [register_next_step_handler][botx.middlewares.ns.register_next_step_handler] function. -We pass into function our message as the first argument and the handler that will be -executed for the next user message as the second. We also can pass key arguments if we need them -and get them in our handler using message state then, but this not our case now. diff --git a/docs/development/handling-errors.md b/docs/development/handling-errors.md deleted file mode 100644 index 464bd21e..00000000 --- a/docs/development/handling-errors.md +++ /dev/null @@ -1,10 +0,0 @@ -`pybotx` provides a mechanism for registering a handler for exceptions that may occur in your command handlers. -By default, these errors are simply logged to the console, but you can register different behavior and perform some actions. -For example, you can handle database disconnection or another runtime errors. You can also use this mechanism to -register the handler for an `Excpetion` error and send info about it to the Sentry with additional information. - -## Usage Example - -```python3 -{!./src/development/handling_errors/handling_errors0.py!} -``` \ No newline at end of file diff --git a/docs/development/logging.md b/docs/development/logging.md deleted file mode 100644 index 450ca097..00000000 --- a/docs/development/logging.md +++ /dev/null @@ -1,7 +0,0 @@ -`pybotx` uses `loguru` internally to log things. - -To enable it, just import `logger` from `loguru` and call `logger.enable("botx")`: - -```Python3 -{!./src/development/logging/logging0.py!} -``` \ No newline at end of file diff --git a/docs/development/sending-data.md b/docs/development/sending-data.md deleted file mode 100644 index 764c6afe..00000000 --- a/docs/development/sending-data.md +++ /dev/null @@ -1,115 +0,0 @@ -`Bot` from `pybotx` provide you 3 methods for sending message to the user (with some additional data) and 1 for sending the file: - -* `.send` - send a message by passing a `SendingMessage`. -* `.answer_message` - send a message by passing text and the original `message` that was passed to the command handler. -* `.send_message` - send message by passing text, `sync_id`, `group_chat_id` or list of them, `bot_id` and `host`. -At most cases you'll prefer `.send` method over this one. -* `.send_file` - send file using file-like object. - -!!! info - Note about using different values to send messages - - - * `sync_id` is the `UUID` accosiated with the message in Express. - You should use it only in command handlers as answer on command or when changing already sent message. - * `group_chat_id` - is the `UUID` accosiated with one of the chats in Express. In most cases, you should use it to - send messages, outside of handlers. - - -### Using `.send` - -`.send` is used to send a message. - -Here is an example of using this method outside from handler: - -```Python3 -{!./src/development/sending_data/sending_data0.py!} -``` - -or inside command handler: - -```Python3 -{!./src/development/sending_data/sending_data1.py!} -``` - -### Using `.answer_message` - -`.answer_message` is very useful for replying to command. - -```Python3 -{!./src/development/sending_data/sending_data2.py!} -``` - -### Send file - -There are several ways to send a file from bot: - -* Attach file to an instance of `SendingMessage`. -* Pass file to `file` argument into `.answer_message` or `.send_message` methods. -* Use `.send_file`. - -#### Attach file to already built message or during initialization - -```Python3 -{!./src/development/sending_data/sending_data3.py!} -``` - -#### Pass file as argument - -```Python3 -{!./src/development/sending_data/sending_data4.py!} -``` - -#### Using `.send_file` - -```Python3 -{!./src/development/sending_data/sending_data5.py!} -``` - -### Attach interactive buttons to your message - -You can attach bubbles or keyboard buttons to your message. This can be done using -`MessageMarkup` class. -A `Bubble` is a button that is stuck to your message. -A `Keyboard` is a panel that will be displayed when -you click on the messege with the icon. - -An attached collection of bubbles or keyboard buttons is a matrix of buttons. - -Adding these elements to your message is pretty easy. -For example, if you want to add 3 buttons to a message (1 in the first line and 2 in the second) -you can do something like this: - -```Python3 -{!./src/development/sending_data/sending_data6.py!} -``` - -Or like this: - -```Python3 -{!./src/development/sending_data/sending_data7.py!} -``` - - -Also you can attach buttons to `SendingMessage` passing it -into `__init__` or after: - -```Python3 -{!./src/development/sending_data/sending_data8.py!} -``` - -### Mention users or another chats in message - -You can mention users or another chats in your messages and they will receive notification -from the chat, even if this chat was muted. - -There are 2 types of mentions for users: - -* Mention user in chat where message will be sent -* Mention just user account - -Here is an example - -```Python3 -{!./src/development/sending_data/sending_data9.py!} -``` diff --git a/docs/development/tests.md b/docs/development/tests.md deleted file mode 100644 index 09e8f2e5..00000000 --- a/docs/development/tests.md +++ /dev/null @@ -1,32 +0,0 @@ -You can test the behaviour of your bot by writing unit tests. Since the main goal of the bot is to process commands and send -results to the BotX API, you should be able to intercept the result between sending data to the API. You can do this by using [TestClient][botx.testing.testing_client.client.TestClient]. -Then you write some mocks and test your logic inside tests. In this example we will `pytest` for unit tests. - -## Example - -### Bot - -Suppose we have a bot that returns a message in the format `"Hello, {username}"` with the command `/hello`: - -`bot.py`: -```python3 -{!./src/development/tests/tests0/bot.py!} -``` - -### Fixtures - -Now let's write some fixtures to use them in our tests: - -`conftest.py`: -```python3 -{!./src/development/tests/tests0/conftest.py!} -``` - -### Tests - -Now we have fixtures for writing tests. Let's write a test to verify that the message body is in the required format: - -`test_format_command.py` -```python3 -{!./src/development/tests/tests0/test_format_command.py!} -``` diff --git a/docs/index.md b/docs/index.md index 56dfe1ec..99408aba 100644 --- a/docs/index.md +++ b/docs/index.md @@ -1,99 +1,5 @@ -

pybotx

-

- A little python framework for building bots for eXpress messenger. -

-

- - Tests - - - Styles - - - Coverage - - - Code Style - - - Package version - - - License - -

+# Минимальный пример бота (интеграция с FastAPI) - ---- - -# Introduction - -`pybotx` is a framework for building bots for eXpress providing a mechanism for simple -integration with your favourite asynchronous web frameworks. - -Main features: - - * Simple integration with your web apps. - * Asynchronous API with synchronous as a fallback option. - * 100% test coverage. - * 100% type annotated codebase. - - -!!! warning - This library is under active development and its API may be unstable. - Please lock the version you are using at the minor update level. For example, like this in `poetry`. - - [tool.poetry.dependencies] - ... - botx = "^0.15.0" - ... - ---- - -## Requirements - -Python 3.7+ - -`pybotx` use the following libraries: - -* pydantic for the data parts. -* httpx for making HTTP calls to BotX API. -* loguru for beautiful and powerful logs. -* **Optional**. Starlette for tests. - -## Installation -```bash -$ pip install botx +``` py +--8<-- "docs/snippets/minimal_example.py" ``` - -Or if you are going to write tests: - -```bash -$ pip install botx[tests] -``` - -You will also need a web framework to create bots as the current BotX API only works with webhooks. -This documentation will use FastAPI for the examples bellow. -```bash -$ pip install fastapi uvicorn -``` - -## Example - -Let's create a simple echo bot. - -* Create a file `main.py` with following content: -```Python3 -{!./src/index/index0.py!} -``` - -* Deploy a bot on your server using uvicorn and set the url for the webhook in Express. -```bash -$ uvicorn main:app --host=0.0.0.0 -``` - -This bot will send back every your message. - -## License - -This project is licensed under the terms of the MIT license. diff --git a/docs/reference/bots.md b/docs/reference/bots.md deleted file mode 100644 index 4da587f1..00000000 --- a/docs/reference/bots.md +++ /dev/null @@ -1,37 +0,0 @@ -::: botx.bots.bots - -::: botx.bots.mixins.collectors - -::: botx.bots.mixins.collecting.handler - -::: botx.bots.mixins.collecting.add_handler - -::: botx.bots.mixins.collecting.default - -::: botx.bots.mixins.collecting.hidden - -::: botx.bots.mixins.collecting.system_events - -::: botx.bots.mixins.exceptions - -::: botx.bots.mixins.lifespan - -::: botx.bots.mixins.middlewares - -::: botx.bots.mixins.sending - -::: botx.bots.mixins.clients - -::: botx.bots.mixins.requests.mixin - -::: botx.bots.mixins.requests.bots - -::: botx.bots.mixins.requests.chats - -::: botx.bots.mixins.requests.command - -::: botx.bots.mixins.requests.events - -::: botx.bots.mixins.requests.notification - -::: botx.bots.mixins.requests.users diff --git a/docs/reference/clients/async-client.md b/docs/reference/clients/async-client.md deleted file mode 100644 index 427183f1..00000000 --- a/docs/reference/clients/async-client.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.clients.clients.async_client diff --git a/docs/reference/clients/methods.md b/docs/reference/clients/methods.md deleted file mode 100644 index 5d4989bc..00000000 --- a/docs/reference/clients/methods.md +++ /dev/null @@ -1,38 +0,0 @@ -## Base Method - -::: botx.clients.methods.base - -## V3 - -## chats - -::: botx.clients.methods.v3.chats.add_admin_role - -::: botx.clients.methods.v3.chats.add_user - -::: botx.clients.methods.v3.chats.create - -::: botx.clients.methods.v3.chats.info - -::: botx.clients.methods.v3.chats.remove_user - -::: botx.clients.methods.v3.chats.stealth_disable - -::: botx.clients.methods.v3.chats.stealth_set - -## command - -::: botx.clients.methods.v3.command.command_result - - -## notification - -::: botx.clients.methods.v3.notification.direct_notification - -# users - -::: botx.clients.methods.v3.users.by_email - -::: botx.clients.methods.v3.users.by_huid - -::: botx.clients.methods.v3.users.by_login diff --git a/docs/reference/clients/sync-client.md b/docs/reference/clients/sync-client.md deleted file mode 100644 index 2f24bd95..00000000 --- a/docs/reference/clients/sync-client.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.clients.clients.sync_client \ No newline at end of file diff --git a/docs/reference/collecting.md b/docs/reference/collecting.md deleted file mode 100644 index 8dcf2fbe..00000000 --- a/docs/reference/collecting.md +++ /dev/null @@ -1,11 +0,0 @@ -::: botx.collecting.collectors.base - -::: botx.collecting.collectors.collector - -::: botx.collecting.collectors.mixins.default - -::: botx.collecting.collectors.mixins.handler - -::: botx.collecting.collectors.mixins.hidden - -::: botx.collecting.collectors.mixins.system_events diff --git a/docs/reference/exceptions.md b/docs/reference/exceptions.md deleted file mode 100644 index 05a55380..00000000 --- a/docs/reference/exceptions.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.exceptions \ No newline at end of file diff --git a/docs/reference/middlewares/authorization.md b/docs/reference/middlewares/authorization.md deleted file mode 100644 index 59fd3123..00000000 --- a/docs/reference/middlewares/authorization.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.middlewares.authorization \ No newline at end of file diff --git a/docs/reference/middlewares/base.md b/docs/reference/middlewares/base.md deleted file mode 100644 index b453947a..00000000 --- a/docs/reference/middlewares/base.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.middlewares.base \ No newline at end of file diff --git a/docs/reference/middlewares/ns.md b/docs/reference/middlewares/ns.md deleted file mode 100644 index d4d6af3e..00000000 --- a/docs/reference/middlewares/ns.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.middlewares.ns \ No newline at end of file diff --git a/docs/reference/models.md b/docs/reference/models.md deleted file mode 100644 index 2ca0f8f2..00000000 --- a/docs/reference/models.md +++ /dev/null @@ -1,39 +0,0 @@ -::: botx.models.buttons - -::: botx.models.chats - -::: botx.models.constants - -::: botx.models.credentials - -::: botx.models.datastructures - -::: botx.models.enums - -::: botx.models.errors - -::: botx.models.events - -::: botx.models.files - -::: botx.models.entities - -::: botx.models.menu - -::: botx.models.messages.incoming_message - -::: botx.models.messages.message - -::: botx.models.messages.sending.credentials - -::: botx.models.messages.sending.markup - -::: botx.models.messages.sending.message - -::: botx.models.messages.sending.options - -::: botx.models.messages.sending.payload - -::: botx.models.typing - -::: botx.models.users diff --git a/docs/reference/testing/message-builder.md b/docs/reference/testing/message-builder.md deleted file mode 100644 index c67c7e13..00000000 --- a/docs/reference/testing/message-builder.md +++ /dev/null @@ -1 +0,0 @@ -::: botx.testing.building.builder \ No newline at end of file diff --git a/docs/reference/testing/test-client.md b/docs/reference/testing/test-client.md deleted file mode 100644 index c177fad8..00000000 --- a/docs/reference/testing/test-client.md +++ /dev/null @@ -1,3 +0,0 @@ -::: botx.testing.testing_client.base - -::: botx.testing.testing_client.client \ No newline at end of file diff --git a/docs/snippets/client/bots_api/get_token.py b/docs/snippets/client/bots_api/get_token.py new file mode 100644 index 00000000..0b4d312c --- /dev/null +++ b/docs/snippets/client/bots_api/get_token.py @@ -0,0 +1,17 @@ +import asyncio + +from botx import Bot, HandlerCollector, lifespan_wrapper + +# Не забудьте заполнить учётные данные бота +built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[]) + + +async def main() -> None: + async with lifespan_wrapper(built_bot) as bot: + for bot_account in bot.bot_accounts: + token = await built_bot.get_token(bot_id=bot_account.id) + print(token) # noqa: WPS421 + + +if __name__ == "__main__": + asyncio.run(main()) diff --git a/docs/snippets/minimal_example.py b/docs/snippets/minimal_example.py new file mode 100644 index 00000000..fba6a750 --- /dev/null +++ b/docs/snippets/minimal_example.py @@ -0,0 +1,72 @@ +from http import HTTPStatus +from uuid import UUID + +from fastapi import FastAPI, Request +from fastapi.responses import JSONResponse + +from botx import ( + Bot, + BotAccountWithSecret, + ChatCreatedEvent, + HandlerCollector, + IncomingMessage, + build_command_accepted_response, +) + +collector = HandlerCollector() + + +@collector.chat_created +async def chat_created_handler(event: ChatCreatedEvent, bot: Bot) -> None: + await bot.answer_message("Hello!") + + +@collector.command("/echo", description="Send back the received message body") +async def echo_handler(message: IncomingMessage, bot: Bot) -> None: + await bot.answer_message(message.body) + + +@collector.default_message_handler +async def default_message_handler(event: IncomingMessage, bot: Bot) -> None: + await bot.answer_message("Sorry, command not found.") + + +bot = Bot( + collectors=[collector], + bot_accounts=[ + BotAccountWithSecret( # noqa: S106 + # Replace fake account credentials with yours + id=UUID("123e4567-e89b-12d3-a456-426655440000"), + host="cts.example.com", + secret_key="e29b417773f2feab9dac143ee3da20c5", + ), + ], +) + +app = FastAPI() +app.add_event_handler("startup", bot.startup) +app.add_event_handler("shutdown", bot.shutdown) + + +@app.post("/command") +async def command_handler(request: Request) -> JSONResponse: + bot.async_execute_raw_bot_command(await request.json()) + return JSONResponse( + build_command_accepted_response(), + status_code=HTTPStatus.ACCEPTED, + ) + + +@app.get("/status") +async def status_handler(request: Request) -> JSONResponse: + status = await bot.raw_get_status(dict(request.query_params)) + return JSONResponse(status) + + +@app.post("/notification/callback") +async def callback_handler(request: Request) -> JSONResponse: + bot.set_raw_botx_method_result(await request.json()) + return JSONResponse( + build_command_accepted_response(), + status_code=HTTPStatus.ACCEPTED, + ) diff --git a/docs/src/development/collector/collector0/a_commands.py b/docs/src/development/collector/collector0/a_commands.py deleted file mode 100644 index 2af9f321..00000000 --- a/docs/src/development/collector/collector0/a_commands.py +++ /dev/null @@ -1,9 +0,0 @@ -from botx import Collector, Message - -collector = Collector() - - -@collector.handler -async def my_handler_for_a_service(message: Message) -> None: - # do something here - print(f"Message from {message.group_chat_id} chat") diff --git a/docs/src/development/collector/collector0/bot.py b/docs/src/development/collector/collector0/bot.py deleted file mode 100644 index 7ba050f1..00000000 --- a/docs/src/development/collector/collector0/bot.py +++ /dev/null @@ -1,11 +0,0 @@ -from botx import Bot - -from .a_commands import collector - -bot = Bot() -bot.include_collector(collector) - - -@bot.default(include_in_status=False) -async def default_handler() -> None: - print("default handler") diff --git a/docs/src/development/dependencies_injection/dependencies_injection0.py b/docs/src/development/dependencies_injection/dependencies_injection0.py deleted file mode 100644 index 42509691..00000000 --- a/docs/src/development/dependencies_injection/dependencies_injection0.py +++ /dev/null @@ -1,14 +0,0 @@ -from uuid import UUID - -from botx import Bot, Depends, Message - -bot = Bot() - - -def get_user_huid(message: Message) -> UUID: - return message.user_huid - - -@bot.handler -async def my_handler(user_huid: UUID = Depends(get_user_huid)) -> None: - print(f"Message from {user_huid}") diff --git a/docs/src/development/dependencies_injection/dependencies_injection1.py b/docs/src/development/dependencies_injection/dependencies_injection1.py deleted file mode 100644 index 382a7df3..00000000 --- a/docs/src/development/dependencies_injection/dependencies_injection1.py +++ /dev/null @@ -1,31 +0,0 @@ -import asyncio -from dataclasses import dataclass -from uuid import UUID - -from botx import Bot, Depends, Message - - -@dataclass -class User: - user_huid: UUID - username: str - - -bot = Bot() - - -def get_user_huid_from_message(message: Message) -> UUID: - return message.user_huid - - -async def fetch_user_by_huid( - user_huid: UUID = Depends(get_user_huid_from_message), -) -> User: - # some operations with db for example - await asyncio.sleep(0.5) - return User(user_huid=user_huid, username="Requested User") - - -@bot.handler -def my_handler(user: User = Depends(fetch_user_by_huid)) -> None: - print(f"Message from {user.username}") diff --git a/docs/src/development/dependencies_injection/dependencies_injection2.py b/docs/src/development/dependencies_injection/dependencies_injection2.py deleted file mode 100644 index a6823be1..00000000 --- a/docs/src/development/dependencies_injection/dependencies_injection2.py +++ /dev/null @@ -1,40 +0,0 @@ -import asyncio -from dataclasses import dataclass -from uuid import UUID - -from botx import Bot, Collector, DependencyFailure, Depends, Message - - -@dataclass -class User: - user_huid: UUID - username: str - is_authenticated: bool - - -collector = Collector() - - -def get_user_huid_from_message(message: Message) -> UUID: - return message.user_huid - - -async def fetch_user_by_huid( - user_huid: UUID = Depends(get_user_huid_from_message), -) -> User: - # some operations with db for example - await asyncio.sleep(0.5) - return User(user_huid=user_huid, username="Requested User", is_authenticated=False) - - -async def authenticate_user( - bot: Bot, message: Message, user: User = Depends(fetch_user_by_huid) -) -> None: - if not user.is_authenticated: - await bot.answer_message("You should login first", message) - raise DependencyFailure - - -@collector.handler(dependencies=[Depends(authenticate_user)]) -def my_handler(user: User = Depends(fetch_user_by_huid)) -> None: - print(f"Message from {user.username}") diff --git a/docs/src/development/first_steps/first_steps0.py b/docs/src/development/first_steps/first_steps0.py deleted file mode 100644 index 57505d76..00000000 --- a/docs/src/development/first_steps/first_steps0.py +++ /dev/null @@ -1,27 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED - -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status - -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) diff --git a/docs/src/development/first_steps/first_steps1.py b/docs/src/development/first_steps/first_steps1.py deleted file mode 100644 index b76b1e3e..00000000 --- a/docs/src/development/first_steps/first_steps1.py +++ /dev/null @@ -1,54 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.requests import Request -from starlette.responses import JSONResponse, Response -from starlette.status import HTTP_202_ACCEPTED, HTTP_503_SERVICE_UNAVAILABLE - -from botx import ( - Bot, - BotDisabledErrorData, - BotDisabledResponse, - BotXCredentials, - IncomingMessage, - Message, - ServerUnknownError, - Status, -) - -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) - - -@app.exception_handler(ServerUnknownError) -async def message_from_unknown_server_hanlder( - _request: Request, exc: ServerUnknownError -) -> Response: - return JSONResponse( - status_code=HTTP_503_SERVICE_UNAVAILABLE, - content=BotDisabledResponse( - error_data=BotDisabledErrorData( - status_message=( - f"Sorry, bot can not communicate with user from {exc.host} CTS" - ), - ), - ).dict(), - ) diff --git a/docs/src/development/first_steps/first_steps2.py b/docs/src/development/first_steps/first_steps2.py deleted file mode 100644 index 09afa4ca..00000000 --- a/docs/src/development/first_steps/first_steps2.py +++ /dev/null @@ -1,45 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED - -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status - -users_data = {} -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -@bot.handler -async def fill_info(message: Message) -> None: - if message.user_huid not in users_data: - text = ( - "Hi! I'm a bot that will ask some questions about you.\n" - "First of all: what is your name?" - ) - else: - text = ( - "You've already filled out information about yourself.\n" - "You can view it by typing `/my-info` command.\n" - "You can also view the processed information by typing `/info` command." - ) - - await bot.answer_message(text, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) diff --git a/docs/src/development/first_steps/first_steps3.py b/docs/src/development/first_steps/first_steps3.py deleted file mode 100644 index 2bdd0fff..00000000 --- a/docs/src/development/first_steps/first_steps3.py +++ /dev/null @@ -1,78 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED - -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status - -users_data = {} -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -@bot.handler -async def fill_info(message: Message) -> None: - if message.user_huid not in users_data: - text = ( - "Hi! I'm a bot that will ask some questions about you.\n" - "First of all: what is your name?" - ) - else: - text = ( - "You've already filled out information about yourself.\n" - "You can view it by typing `/my-info` command.\n" - "You can also view the processed information by typing `/info` command." - ) - - await bot.answer_message(text, message) - - -@bot.handler(command="/my-info") -async def get_info_for_user(message: Message) -> None: - if message.user_huid not in users_data: - text = ( - "I have no information about you :(\n" - "Type `/fill-info` so I can collect it, please." - ) - await bot.answer_message(text, message) - else: - text = ( - f"Your name: {users_data[message.user_huid]['name']}\n" - f"Your age: {users_data[message.user_huid]['age']}\n" - f"Your gender: {users_data[message.user_huid]['gender']}\n" - "This is all that I have now." - ) - await bot.answer_message(text, message) - - -@bot.handler(commands=["/info", "/information"]) -async def get_processed_information(message: Message) -> None: - users_count = len(users_data) - average_age = sum(user["age"] for user in users_data) / users_count - gender_array = [1 if user["gender"] == "male" else 2 for user in users_data] - text = ( - f"Count of users: {users_count}\n" - f"Average age: {average_age}\n" - f"Male users count: {gender_array.count(1)}\n" - f"Female users count: {gender_array.count(2)}" - ) - - await bot.answer_message(text, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) diff --git a/docs/src/development/first_steps/first_steps4.py b/docs/src/development/first_steps/first_steps4.py deleted file mode 100644 index 69bb994c..00000000 --- a/docs/src/development/first_steps/first_steps4.py +++ /dev/null @@ -1,126 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED - -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status -from botx.middlewares.ns import NextStepMiddleware, register_next_step_handler - -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - -users_data = {} - - -async def get_name(message: Message) -> None: - users_data[message.user_huid]["name"] = message.body - await bot.answer_message("Good! Move next: how old are you?", message) - register_next_step_handler(message, get_age) - - -async def get_age(message: Message) -> None: - try: - age = int(message.body) - if age <= 2: - await bot.answer_message( - "Sorry, but it's not true. Say your real age, please!", message, - ) - register_next_step_handler(message, get_age) - else: - users_data[message.user_huid]["age"] = age - await bot.answer_message("Got it! Final question: your gender?", message) - register_next_step_handler(message, get_gender) - except ValueError: - await bot.answer_message( - "No, no, no. Pleas tell me your age in numbers!", message, - ) - register_next_step_handler(message, get_age) - - -async def get_gender(message: Message) -> None: - gender = message.body - if gender in ["male", "female"]: - users_data[message.user_huid]["gender"] = gender - await bot.answer_message( - "Ok! Thanks for taking the time to answer my questions.", message, - ) - else: - await bot.answer_message( - "Sorry, but I can not recognize your answer! Type 'male' or 'female', please!", - message, - ) - register_next_step_handler(message, get_gender) - - -bot.add_middleware( - NextStepMiddleware, bot=bot, functions={get_age, get_name, get_gender}, -) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -@bot.handler -async def fill_info(message: Message) -> None: - if message.user_huid not in users_data: - text = ( - "Hi! I'm a bot that will ask some questions about you.\n" - "First of all: what is your name?" - ) - register_next_step_handler(message, get_name) - else: - text = ( - "You've already filled out information about yourself.\n" - "You can view it by typing `/my-info` command.\n" - "You can also view the processed information by typing `/info` command." - ) - - await bot.answer_message(text, message) - - -@bot.handler(command="/my-info") -async def get_info_for_user(message: Message) -> None: - if message.user_huid not in users_data: - text = ( - "I have no information about you :(\n" - "Type `/fill-info` so I can collect it, please." - ) - await bot.answer_message(text, message) - else: - text = ( - f"Your name: {users_data[message.user_huid]['name']}\n" - f"Your age: {users_data[message.user_huid]['age']}\n" - f"Your gender: {users_data[message.user_huid]['gender']}\n" - "This is all that I have now." - ) - await bot.answer_message(text, message) - - -@bot.handler(commands=["/info", "/information"]) -async def get_processed_information(message: Message) -> None: - users_count = len(users_data) - average_age = sum(user["age"] for user in users_data) / users_count - gender_array = [1 if user["gender"] == "male" else 2 for user in users_data] - text = ( - f"Count of users: {users_count}\n" - f"Average age: {average_age}\n" - f"Male users count: {gender_array.count(1)}\n" - f"Female users count: {gender_array.count(2)}" - ) - - await bot.answer_message(text, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) diff --git a/docs/src/development/handling_errors/handling_errors0.py b/docs/src/development/handling_errors/handling_errors0.py deleted file mode 100644 index 41cbd170..00000000 --- a/docs/src/development/handling_errors/handling_errors0.py +++ /dev/null @@ -1,13 +0,0 @@ -from botx import Bot, Message - -bot = Bot() - - -@bot.exception_handler(RuntimeError) -async def error_handler(exc: Exception, msg: Message) -> None: - await msg.bot.answer_message(f"Error occurred during handling command: {exc}", msg) - - -@bot.handler -async def handler_with_bug(message: Message) -> None: - raise RuntimeError(message.body) diff --git a/docs/src/development/logging/logging0.py b/docs/src/development/logging/logging0.py deleted file mode 100644 index ac71a13e..00000000 --- a/docs/src/development/logging/logging0.py +++ /dev/null @@ -1,6 +0,0 @@ -from loguru import logger - -from botx import Bot - -bot = Bot() -logger.enable("botx") diff --git a/docs/src/development/sending_data/sending_data0.py b/docs/src/development/sending_data/sending_data0.py deleted file mode 100644 index b9b60d6a..00000000 --- a/docs/src/development/sending_data/sending_data0.py +++ /dev/null @@ -1,18 +0,0 @@ -from uuid import UUID - -from botx import Bot, SendingMessage - -bot = Bot() -CHAT_ID = UUID("1f972f5e-6d17-4f39-be5b-f7e20f1b4d13") -BOT_ID = UUID("cc257e1c-c028-4181-a055-01e14ba881b0") -CTS_HOST = "my-cts.example.com" - - -async def some_function() -> None: - message = SendingMessage( - text="You were chosen by random.", - bot_id=BOT_ID, - host=CTS_HOST, - chat_id=CHAT_ID, - ) - await bot.send(message) diff --git a/docs/src/development/sending_data/sending_data1.py b/docs/src/development/sending_data/sending_data1.py deleted file mode 100644 index dfe95fb3..00000000 --- a/docs/src/development/sending_data/sending_data1.py +++ /dev/null @@ -1,11 +0,0 @@ -from botx import Bot, Message, SendingMessage - -bot = Bot() - - -@bot.handler(command="/my-handler") -async def some_handler(message: Message) -> None: - message = SendingMessage.from_message( - text="You were chosen by random.", message=message, - ) - await bot.send(message) diff --git a/docs/src/development/sending_data/sending_data2.py b/docs/src/development/sending_data/sending_data2.py deleted file mode 100644 index 0532089f..00000000 --- a/docs/src/development/sending_data/sending_data2.py +++ /dev/null @@ -1,8 +0,0 @@ -from botx import Bot, Message - -bot = Bot() - - -@bot.handler(command="/my-handler") -async def some_handler(message: Message) -> None: - await bot.answer_message(text="VERY IMPORTANT NOTIFICATION!!!", message=message) diff --git a/docs/src/development/sending_data/sending_data3.py b/docs/src/development/sending_data/sending_data3.py deleted file mode 100644 index 5d0c4ac8..00000000 --- a/docs/src/development/sending_data/sending_data3.py +++ /dev/null @@ -1,23 +0,0 @@ -from botx import Bot, File, Message, SendingMessage - -bot = Bot() - - -@bot.handler -async def my_handler(message: Message) -> None: - with open("my_file.txt") as f: - notification = SendingMessage( - file=File.from_file(f), credentials=message.credentials, - ) - - await bot.send(notification) - - -@bot.handler -async def another_handler(message: Message) -> None: - notification = SendingMessage.from_message(message=message) - - with open("my_file.txt") as f: - notification.add_file(f) - - await bot.send(notification) diff --git a/docs/src/development/sending_data/sending_data4.py b/docs/src/development/sending_data/sending_data4.py deleted file mode 100644 index 576425ec..00000000 --- a/docs/src/development/sending_data/sending_data4.py +++ /dev/null @@ -1,9 +0,0 @@ -from botx import Bot, Message - -bot = Bot() - - -@bot.handler -async def my_handler(message: Message) -> None: - with open("my_file.txt") as f: - await bot.answer_message("Text that will be sent with file", message, file=f) diff --git a/docs/src/development/sending_data/sending_data5.py b/docs/src/development/sending_data/sending_data5.py deleted file mode 100644 index 8c3dc128..00000000 --- a/docs/src/development/sending_data/sending_data5.py +++ /dev/null @@ -1,9 +0,0 @@ -from botx import Bot, Message - -bot = Bot() - - -@bot.handler -async def my_handler(message: Message) -> None: - with open("my_file.txt") as f: - await bot.send_file(f, message.credentials) diff --git a/docs/src/development/sending_data/sending_data6.py b/docs/src/development/sending_data/sending_data6.py deleted file mode 100644 index 3c0c08b2..00000000 --- a/docs/src/development/sending_data/sending_data6.py +++ /dev/null @@ -1,20 +0,0 @@ -from botx import Bot, BubbleElement, Message, MessageMarkup - -bot = Bot() - - -@bot.handler -async def my_handler_with_direct_bubbles_definition(message: Message) -> None: - await bot.answer_message( - "Bubbles!!", - message, - markup=MessageMarkup( - bubbles=[ - [BubbleElement(label="bubble 1", command="")], - [ - BubbleElement(label="bubble 2", command=""), - BubbleElement(label="bubble 3", command=""), - ], - ], - ), - ) diff --git a/docs/src/development/sending_data/sending_data7.py b/docs/src/development/sending_data/sending_data7.py deleted file mode 100644 index 55be109d..00000000 --- a/docs/src/development/sending_data/sending_data7.py +++ /dev/null @@ -1,13 +0,0 @@ -from botx import Bot, Message, MessageMarkup - -bot = Bot() - - -@bot.handler -async def my_handler_with_passing_predefined_markup(message: Message) -> None: - markup = MessageMarkup() - markup.add_bubble(command="", label="bubble 1") - markup.add_bubble(command="", label="bubble 2", new_row=False) - markup.add_bubble(command="", label="bubble 3") - - await bot.answer_message("Bubbles!!", message, markup=markup) diff --git a/docs/src/development/sending_data/sending_data8.py b/docs/src/development/sending_data/sending_data8.py deleted file mode 100644 index 540bda91..00000000 --- a/docs/src/development/sending_data/sending_data8.py +++ /dev/null @@ -1,13 +0,0 @@ -from botx import Bot, Message, SendingMessage - -bot = Bot() - - -@bot.handler -async def my_handler_with_markup_in_sending_message(message: Message) -> None: - reply = SendingMessage.from_message(text="More buttons!!!", message=message) - reply.add_bubble(command="", label="bubble 1") - reply.add_keyboard_button(command="", label="keyboard button 1", new_row=False) - reply.add_keyboard_button(command="", label="keyboard button 2") - - await bot.send(reply) diff --git a/docs/src/development/sending_data/sending_data9.py b/docs/src/development/sending_data/sending_data9.py deleted file mode 100644 index 0d1a2e08..00000000 --- a/docs/src/development/sending_data/sending_data9.py +++ /dev/null @@ -1,33 +0,0 @@ -from uuid import UUID - -from botx import Bot, Message, SendingMessage - -bot = Bot() -CHAT_FOR_MENTION = UUID("369b49fd-b5eb-4d5b-8e4d-83b020ff2b14") -USER_FOR_MENTION = UUID("cbf4b952-77d5-4484-aea0-f05fb622e089") - - -@bot.handler -async def my_handler_with_user_mention(message: Message) -> None: - reply = SendingMessage.from_message( - text="Hi! There is a notification with mention for you", message=message, - ) - reply.mention_user(message.user_huid) - - await bot.send(reply) - - -@bot.handler -async def my_handler_with_chat_mention(message: Message) -> None: - reply = SendingMessage.from_message(text="Check this chat", message=message) - reply.mention_chat(CHAT_FOR_MENTION, name="Interesting chat") - await bot.send(reply) - - -@bot.handler -async def my_handler_with_contact_mention(message: Message) -> None: - reply = SendingMessage.from_message( - text="You should request access!", message=message, - ) - reply.mention_chat(USER_FOR_MENTION, name="Administrator") - await bot.send(reply) diff --git a/docs/src/development/tests/tests0/bot.py b/docs/src/development/tests/tests0/bot.py deleted file mode 100644 index 03aa1ae1..00000000 --- a/docs/src/development/tests/tests0/bot.py +++ /dev/null @@ -1,8 +0,0 @@ -from botx import Bot, Message - -bot = Bot() - - -@bot.handler -async def hello(message: Message) -> None: - await bot.answer_message(f"Hello, {message.user.username}", message) diff --git a/docs/src/development/tests/tests0/conftest.py b/docs/src/development/tests/tests0/conftest.py deleted file mode 100644 index 6bdd3f52..00000000 --- a/docs/src/development/tests/tests0/conftest.py +++ /dev/null @@ -1,26 +0,0 @@ -from uuid import UUID - -import pytest - -from botx import Bot, BotXCredentials, MessageBuilder, TestClient - -from .bot import bot - - -@pytest.fixture -def builder() -> MessageBuilder: - builder = MessageBuilder() - builder.user.host = "example.com" - return builder - - -@pytest.fixture -def bot(builder: MessageBuilder) -> Bot: - bot.bot_accounts.append(BotXCredentials(host=builder.user.host, secret_key="secret", bot_id=UUID("bot_id"))) - return bot - - -@pytest.fixture -def client(bot: Bot) -> TestClient: - with TestClient(bot) as client: - yield client diff --git a/docs/src/development/tests/tests0/test_format_command.py b/docs/src/development/tests/tests0/test_format_command.py deleted file mode 100644 index cec31a30..00000000 --- a/docs/src/development/tests/tests0/test_format_command.py +++ /dev/null @@ -1,15 +0,0 @@ -import pytest - -from botx import Bot, MessageBuilder, TestClient - - -@pytest.mark.asyncio -async def test_hello_format( - bot: Bot, builder: MessageBuilder, client: TestClient -) -> None: - builder.body = "/hello" - - await client.send_command(builder.message) - - command_result = client.command_results[0] - assert command_result.result.body == f"Hello, {builder.user.username}" diff --git a/docs/src/index/index0.py b/docs/src/index/index0.py deleted file mode 100644 index 57505d76..00000000 --- a/docs/src/index/index0.py +++ /dev/null @@ -1,27 +0,0 @@ -from uuid import UUID - -from fastapi import FastAPI -from starlette.status import HTTP_202_ACCEPTED - -from botx import Bot, BotXCredentials, IncomingMessage, Message, Status - -bot = Bot(bot_accounts=[BotXCredentials(host="cts.example.com", secret_key="secret", bot_id=UUID("bot_id"))]) - - -@bot.default(include_in_status=False) -async def echo_handler(message: Message) -> None: - await bot.answer_message(message.body, message) - - -app = FastAPI() -app.add_event_handler("shutdown", bot.shutdown) - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command", status_code=HTTP_202_ACCEPTED) -async def bot_command(message: IncomingMessage) -> None: - await bot.execute_command(message.dict()) diff --git a/examples/README.md b/examples/README.md deleted file mode 100644 index 6056ec49..00000000 --- a/examples/README.md +++ /dev/null @@ -1,8 +0,0 @@ -# Examples - -In this folder are small examples to show different ways of usage `pybotx` library for -writing bots. - -### [Finite-state machine middleware](./fsm) - -An example of bot and middleware that defines finite-state machine behaviour. \ No newline at end of file diff --git a/examples/fsm/.env.example b/examples/fsm/.env.example deleted file mode 100644 index 704e6987..00000000 --- a/examples/fsm/.env.example +++ /dev/null @@ -1,2 +0,0 @@ -CTS_HOST=cts.example.com -BOT_SECRET=secret key \ No newline at end of file diff --git a/examples/fsm/README.md b/examples/fsm/README.md deleted file mode 100644 index 6e368143..00000000 --- a/examples/fsm/README.md +++ /dev/null @@ -1,15 +0,0 @@ -## General - -A bot that defines middleware that process request in finite-state machine way. - -State is an enum (`enum.Enum`) with several values that are changed using -`bot.middleware.change_state` function. - -This example shows definition of custom middleware and handlers processing logic. - -## Run - -start `uvicorn` ASGI server with bot on 8000 port with following command: -```bash -$ uvicorn bot.web:app -``` \ No newline at end of file diff --git a/examples/fsm/bot/bot.py b/examples/fsm/bot/bot.py deleted file mode 100644 index 0777a586..00000000 --- a/examples/fsm/bot/bot.py +++ /dev/null @@ -1,25 +0,0 @@ -from typing import Any -from uuid import UUID - -from bot.config import BOT_SECRET, CTS_HOST -from bot.handlers import FSMStates, fsm -from bot.middleware import FlowError, FSMMiddleware, change_state -from botx import Bot, BotXCredentials, Message - -bot = Bot(bot_accounts=[BotXCredentials(host=CTS_HOST, secret_key=str(BOT_SECRET), bot_id=UUID("bot_id"))]) -bot.add_middleware(FSMMiddleware, bot=bot, fsm=fsm) - - -@bot.default(include_in_status=False) -async def default_handler(message: Message) -> None: - if message.body == "start": - change_state(message, FSMStates.get_first_name) - await message.bot.answer_message("enter first name", message) - return - - await message.bot.answer_message("default handler", message) - - -@bot.exception_handler(FlowError) -async def flow_error_handler(*_: Any) -> None: - pass diff --git a/examples/fsm/bot/config.py b/examples/fsm/bot/config.py deleted file mode 100644 index dfd190e5..00000000 --- a/examples/fsm/bot/config.py +++ /dev/null @@ -1,10 +0,0 @@ -from loguru import logger -from starlette.config import Config -from starlette.datastructures import Secret - -config = Config(".env") - -CTS_HOST: str = config("CTS_HOST") -BOT_SECRET: Secret = config("BOT_SECRET", cast=Secret) - -logger.enable("botx") diff --git a/examples/fsm/bot/handlers.py b/examples/fsm/bot/handlers.py deleted file mode 100644 index ab3217b5..00000000 --- a/examples/fsm/bot/handlers.py +++ /dev/null @@ -1,60 +0,0 @@ -import pprint -from collections import defaultdict -from enum import Enum, auto -from typing import DefaultDict, Dict -from uuid import UUID - -from bot.middleware import FSM, FlowError -from botx import Message - - -class FSMStates(Enum): - get_first_name = auto() - get_middle_name = auto() - get_last_name = auto() - get_age = auto() - get_gender = auto() - end = auto() - - -fsm = FSM(FSMStates) -user_info: DefaultDict[UUID, Dict[str, str]] = defaultdict(dict) - - -@fsm.handler(on_state=FSMStates.get_first_name, next_state=FSMStates.get_middle_name) -async def get_first_name(message: Message) -> None: - user_info[message.user_huid]["first_name"] = message.body - await message.bot.answer_message("enter middle name", message) - - -@fsm.handler(on_state=FSMStates.get_middle_name, next_state=FSMStates.get_last_name) -async def get_middle_name(message: Message) -> None: - user_info[message.user_huid]["middle_name"] = message.body - await message.bot.answer_message("enter last name", message) - - -@fsm.handler( - on_state=FSMStates.get_last_name, - next_state=FSMStates.get_age, - on_failure=FSMStates.get_first_name, -) -async def get_last_name(message: Message) -> None: - if message.body == "fail": - await message.bot.answer_message("failed to read last name", message) - raise FlowError - - await message.bot.answer_message("enter age", message) - user_info[message.user_huid]["last_name"] = message.body - - -@fsm.handler(on_state=FSMStates.get_age, next_state=FSMStates.get_gender) -async def get_age(message: Message) -> None: - await message.bot.answer_message("enter gender", message) - user_info[message.user_huid]["age"] = message.body - - -@fsm.handler(on_state=FSMStates.get_gender, next_state=None) -async def get_gender(message: Message) -> None: - user_info[message.user_huid]["gender"] = message.body - await message.bot.answer_message("thanks for sharing info:", message) - await message.bot.answer_message(pprint.pformat(user_info), message) diff --git a/examples/fsm/bot/middleware.py b/examples/fsm/bot/middleware.py deleted file mode 100644 index b485f685..00000000 --- a/examples/fsm/bot/middleware.py +++ /dev/null @@ -1,88 +0,0 @@ -from dataclasses import dataclass -from enum import Enum -from typing import Callable, Dict, Final, Optional, Type, Union - -from botx import Bot, Collector, Message -from botx.concurrency import callable_to_coroutine -from botx.middlewares.base import BaseMiddleware -from botx.typing import Executor - -_default_transition: Final = object() - - -@dataclass -class Transition: - on_failure: Optional[Union[Enum, object]] = _default_transition - on_success: Optional[Union[Enum, object]] = _default_transition - - -class FlowError(Exception): - pass - - -class FSM: - def __init__(self, states: Type[Enum]) -> None: - self.transitions: Dict[Enum, Transition] = {} - self.collector = Collector() - self.states = states - - def handler( - self, - on_state: Enum, - next_state: Optional[Union[Enum, object]] = _default_transition, - on_failure: Optional[Union[Enum, object]] = _default_transition, - ) -> Callable: - def decorator(handler: Callable) -> Callable: - self.collector.add_handler( - handler, - body=on_state.name, - name=on_state.name, - include_in_status=False, - ) - self.transitions[on_state] = Transition( - on_success=next_state, on_failure=on_failure, - ) - - return handler - - return decorator - - -def change_state(message: Message, new_state: Optional[Enum]) -> None: - message.bot.state.fsm_state[(message.user_huid, message.group_chat_id)] = new_state - - -class FSMMiddleware(BaseMiddleware): - def __init__( - self, - executor: Executor, - bot: Bot, - fsm: FSM, - initial_state: Optional[Enum] = None, - ) -> None: - super().__init__(executor) - bot.state.fsm_state = {} - self.fsm = fsm - self.initial_state = initial_state - for state in self.fsm.states: - # check that for each state there is registered handler - assert state in self.fsm.transitions - - async def dispatch(self, message: Message, call_next: Executor) -> None: - current_state: Enum = message.bot.state.fsm_state.setdefault( - (message.user_huid, message.group_chat_id), self.initial_state, - ) - if current_state is not None: - transition = self.fsm.transitions[current_state] - handler = self.fsm.collector.handler_for(current_state.name) - try: - await handler(message) - except Exception as exc: - if transition.on_failure is not _default_transition: - change_state(message, transition.on_failure) - raise exc - else: - if transition.on_success is not _default_transition: - change_state(message, transition.on_success) - else: - await callable_to_coroutine(call_next, message) diff --git a/examples/fsm/bot/web.py b/examples/fsm/bot/web.py deleted file mode 100644 index b1117420..00000000 --- a/examples/fsm/bot/web.py +++ /dev/null @@ -1,16 +0,0 @@ -from fastapi import FastAPI - -from bot.bot import bot -from botx import Status - -app = FastAPI() - - -@app.get("/status", response_model=Status) -async def bot_status() -> Status: - return await bot.status() - - -@app.post("/command") -async def bot_command(message: dict) -> None: - await bot.execute_command(message) diff --git a/examples/fsm/poetry.lock b/examples/fsm/poetry.lock deleted file mode 100644 index 7c3c5273..00000000 --- a/examples/fsm/poetry.lock +++ /dev/null @@ -1,599 +0,0 @@ -[[package]] -category = "dev" -description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." -name = "appdirs" -optional = false -python-versions = "*" -version = "1.4.3" - -[[package]] -category = "dev" -description = "Classes Without Boilerplate" -name = "attrs" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "19.3.0" - -[package.extras] -azure-pipelines = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "pytest-azurepipelines"] -dev = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface", "sphinx", "pre-commit"] -docs = ["sphinx", "zope.interface"] -tests = ["coverage", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "zope.interface"] - -[[package]] -category = "dev" -description = "Removes unused imports and unused variables" -name = "autoflake" -optional = false -python-versions = "*" -version = "1.3.1" - -[package.dependencies] -pyflakes = ">=1.1.0" - -[[package]] -category = "dev" -description = "The uncompromising code formatter." -name = "black" -optional = false -python-versions = ">=3.6" -version = "19.10b0" - -[package.dependencies] -appdirs = "*" -attrs = ">=18.1.0" -click = ">=6.5" -pathspec = ">=0.6,<1" -regex = "*" -toml = ">=0.9.4" -typed-ast = ">=1.4.0" - -[package.extras] -d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] - -[[package]] -category = "main" -description = "A little python framework for building bots for eXpress" -name = "botx" -optional = false -python-versions = ">=3.6,<4.0" -version = "0.13.3" - -[package.dependencies] -httpx = ">=0.11.0,<0.12.0" -loguru = ">=0.4.0,<0.5.0" -pydantic = ">=1.0,<2.0" - -[package.extras] -docs = ["mkdocs (>=1.0,<2.0)", "mkdocs-material (>=4.4,<5.0)", "mkdocstrings (>=0.7,<0.8)", "markdown-include (>=0.5.1,<0.6.0)", "fastapi (>=0.47.0,<0.48.0)"] -tests = ["starlette (>=0.12.9,<0.13.0)"] - -[[package]] -category = "main" -description = "Python package for providing Mozilla's CA Bundle." -name = "certifi" -optional = false -python-versions = "*" -version = "2019.11.28" - -[[package]] -category = "main" -description = "Universal encoding detector for Python 2 and 3" -name = "chardet" -optional = false -python-versions = "*" -version = "3.0.4" - -[[package]] -category = "main" -description = "Composable command line interface toolkit" -name = "click" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "7.0" - -[[package]] -category = "main" -description = "Cross-platform colored terminal text." -marker = "sys_platform == \"win32\"" -name = "colorama" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" -version = "0.4.3" - -[[package]] -category = "main" -description = "FastAPI framework, high performance, easy to learn, fast to code, ready for production" -name = "fastapi" -optional = false -python-versions = ">=3.6" -version = "0.48.0" - -[package.dependencies] -pydantic = ">=0.32.2,<2.0.0" -starlette = "0.12.9" - -[package.extras] -all = ["requests", "aiofiles", "jinja2", "python-multipart", "itsdangerous", "pyyaml", "graphene", "ujson", "email-validator", "uvicorn", "async-exit-stack", "async-generator"] -dev = ["pyjwt", "passlib", "autoflake", "flake8", "uvicorn", "graphene"] -doc = ["mkdocs", "mkdocs-material", "markdown-include"] -test = ["pytest (>=4.0.0)", "pytest-cov", "mypy", "black", "isort", "requests", "email-validator", "sqlalchemy", "peewee", "databases", "orjson", "async-exit-stack", "async-generator"] - -[[package]] -category = "main" -description = "A pure-Python, bring-your-own-I/O implementation of HTTP/1.1" -name = "h11" -optional = false -python-versions = "*" -version = "0.9.0" - -[[package]] -category = "main" -description = "HTTP/2 State-Machine based protocol implementation" -name = "h2" -optional = false -python-versions = "*" -version = "3.2.0" - -[package.dependencies] -hpack = ">=3.0,<4" -hyperframe = ">=5.2.0,<6" - -[[package]] -category = "main" -description = "Pure-Python HPACK header compression" -name = "hpack" -optional = false -python-versions = "*" -version = "3.0.0" - -[[package]] -category = "main" -description = "Chromium HSTS Preload list as a Python package and updated daily" -name = "hstspreload" -optional = false -python-versions = ">=3.6" -version = "2020.2.25" - -[[package]] -category = "main" -description = "A collection of framework independent HTTP protocol utils." -marker = "sys_platform != \"win32\" and sys_platform != \"cygwin\" and platform_python_implementation != \"PyPy\"" -name = "httptools" -optional = false -python-versions = "*" -version = "0.1.1" - -[package.extras] -test = ["Cython (0.29.14)"] - -[[package]] -category = "main" -description = "The next generation HTTP client." -name = "httpx" -optional = false -python-versions = ">=3.6" -version = "0.11.1" - -[package.dependencies] -certifi = "*" -chardet = ">=3.0.0,<4.0.0" -h11 = ">=0.8,<0.10" -h2 = ">=3.0.0,<4.0.0" -hstspreload = "*" -idna = ">=2.0.0,<3.0.0" -rfc3986 = ">=1.3,<2" -sniffio = ">=1.0.0,<2.0.0" -urllib3 = ">=1.0.0,<2.0.0" - -[[package]] -category = "main" -description = "HTTP/2 framing layer for Python" -name = "hyperframe" -optional = false -python-versions = "*" -version = "5.2.0" - -[[package]] -category = "main" -description = "Internationalized Domain Names in Applications (IDNA)" -name = "idna" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "2.9" - -[[package]] -category = "dev" -description = "A Python utility / library to sort Python imports." -name = "isort" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "4.3.21" - -[package.extras] -pipfile = ["pipreqs", "requirementslib"] -pyproject = ["toml"] -requirements = ["pipreqs", "pip-api"] -xdg_home = ["appdirs (>=1.4.0)"] - -[[package]] -category = "main" -description = "Python logging made (stupidly) simple" -name = "loguru" -optional = false -python-versions = ">=3.5" -version = "0.4.1" - -[package.dependencies] -colorama = ">=0.3.4" -win32-setctime = ">=1.0.0" - -[package.extras] -dev = ["codecov (>=2.0.15)", "colorama (>=0.3.4)", "flake8 (>=3.7.7)", "isort (>=4.3.20)", "tox (>=3.9.0)", "tox-travis (>=0.12)", "pytest (>=4.6.2)", "pytest-cov (>=2.7.1)", "Sphinx (>=2.2.1)", "sphinx-autobuild (>=0.7.1)", "sphinx-rtd-theme (>=0.4.3)", "black (>=19.3b0)"] - -[[package]] -category = "dev" -description = "Utility library for gitignore style pattern matching of file paths." -name = "pathspec" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" -version = "0.7.0" - -[[package]] -category = "main" -description = "Data validation and settings management using python 3.6 type hinting" -name = "pydantic" -optional = false -python-versions = ">=3.6" -version = "1.4" - -[package.extras] -dotenv = ["python-dotenv (>=0.10.4)"] -email = ["email-validator (>=1.0.3)"] -typing_extensions = ["typing-extensions (>=3.7.2)"] - -[[package]] -category = "dev" -description = "passive checker of Python programs" -name = "pyflakes" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" -version = "2.1.1" - -[[package]] -category = "dev" -description = "Alternative regular expression module, to replace re." -name = "regex" -optional = false -python-versions = "*" -version = "2020.2.20" - -[[package]] -category = "main" -description = "Validating URI References per RFC 3986" -name = "rfc3986" -optional = false -python-versions = "*" -version = "1.3.2" - -[package.extras] -idna2008 = ["idna"] - -[[package]] -category = "main" -description = "Sniff out which async library your code is running under" -name = "sniffio" -optional = false -python-versions = ">=3.5" -version = "1.1.0" - -[[package]] -category = "main" -description = "The little ASGI library that shines." -name = "starlette" -optional = false -python-versions = ">=3.6" -version = "0.12.9" - -[package.extras] -full = ["aiofiles", "graphene", "itsdangerous", "jinja2", "python-multipart", "pyyaml", "requests", "ujson"] - -[[package]] -category = "dev" -description = "Python Library for Tom's Obvious, Minimal Language" -name = "toml" -optional = false -python-versions = "*" -version = "0.10.0" - -[[package]] -category = "dev" -description = "a fork of Python 2 and 3 ast modules with type comment support" -name = "typed-ast" -optional = false -python-versions = "*" -version = "1.4.1" - -[[package]] -category = "main" -description = "HTTP library with thread-safe connection pooling, file post, and more." -name = "urllib3" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" -version = "1.25.8" - -[package.extras] -brotli = ["brotlipy (>=0.6.0)"] -secure = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "certifi", "ipaddress"] -socks = ["PySocks (>=1.5.6,<1.5.7 || >1.5.7,<2.0)"] - -[[package]] -category = "main" -description = "The lightning-fast ASGI server." -name = "uvicorn" -optional = false -python-versions = "*" -version = "0.11.3" - -[package.dependencies] -click = ">=7.0.0,<8.0.0" -h11 = ">=0.8,<0.10" -httptools = ">=0.1.0,<0.2.0" -uvloop = ">=0.14.0" -websockets = ">=8.0.0,<9.0.0" - -[[package]] -category = "main" -description = "Fast implementation of asyncio event loop on top of libuv" -marker = "sys_platform != \"win32\" and sys_platform != \"cygwin\" and platform_python_implementation != \"PyPy\"" -name = "uvloop" -optional = false -python-versions = "*" -version = "0.14.0" - -[[package]] -category = "main" -description = "An implementation of the WebSocket Protocol (RFC 6455 & 7692)" -name = "websockets" -optional = false -python-versions = ">=3.6.1" -version = "8.1" - -[[package]] -category = "main" -description = "A small Python utility to set file creation time on Windows" -marker = "sys_platform == \"win32\"" -name = "win32-setctime" -optional = false -python-versions = ">=3.5" -version = "1.0.1" - -[package.extras] -dev = ["pytest (>=4.6.2)", "black (>=19.3b0)"] - -[metadata] -content-hash = "e6eb201134d6b1d79b81049d96a3cf895462598d51cfb9438a423f5c4033600b" -python-versions = "^3.8" - -[metadata.files] -appdirs = [ - {file = "appdirs-1.4.3-py2.py3-none-any.whl", hash = "sha256:d8b24664561d0d34ddfaec54636d502d7cea6e29c3eaf68f3df6180863e2166e"}, - {file = "appdirs-1.4.3.tar.gz", hash = "sha256:9e5896d1372858f8dd3344faf4e5014d21849c756c8d5701f78f8a103b372d92"}, -] -attrs = [ - {file = "attrs-19.3.0-py2.py3-none-any.whl", hash = "sha256:08a96c641c3a74e44eb59afb61a24f2cb9f4d7188748e76ba4bb5edfa3cb7d1c"}, - {file = "attrs-19.3.0.tar.gz", hash = "sha256:f7b7ce16570fe9965acd6d30101a28f62fb4a7f9e926b3bbc9b61f8b04247e72"}, -] -autoflake = [ - {file = "autoflake-1.3.1.tar.gz", hash = "sha256:680cb9dade101ed647488238ccb8b8bfb4369b53d58ba2c8cdf7d5d54e01f95b"}, -] -black = [ - {file = "black-19.10b0-py36-none-any.whl", hash = "sha256:1b30e59be925fafc1ee4565e5e08abef6b03fe455102883820fe5ee2e4734e0b"}, - {file = "black-19.10b0.tar.gz", hash = "sha256:c2edb73a08e9e0e6f65a0e6af18b059b8b1cdd5bef997d7a0b181df93dc81539"}, -] -botx = [ - {file = "botx-0.13.3-py3-none-any.whl", hash = "sha256:a87768a47105c699ad26bae1a5b930e360ccc792940f3f22ead3e80da2c7a158"}, - {file = "botx-0.13.3.tar.gz", hash = "sha256:b7a68a359c793a2ebaa0c7b86aa7284c997d5cf8bb0c3c3b1987337b86bb9c0c"}, -] -certifi = [ - {file = "certifi-2019.11.28-py2.py3-none-any.whl", hash = "sha256:017c25db2a153ce562900032d5bc68e9f191e44e9a0f762f373977de9df1fbb3"}, - {file = "certifi-2019.11.28.tar.gz", hash = "sha256:25b64c7da4cd7479594d035c08c2d809eb4aab3a26e5a990ea98cc450c320f1f"}, -] -chardet = [ - {file = "chardet-3.0.4-py2.py3-none-any.whl", hash = "sha256:fc323ffcaeaed0e0a02bf4d117757b98aed530d9ed4531e3e15460124c106691"}, - {file = "chardet-3.0.4.tar.gz", hash = "sha256:84ab92ed1c4d4f16916e05906b6b75a6c0fb5db821cc65e70cbd64a3e2a5eaae"}, -] -click = [ - {file = "Click-7.0-py2.py3-none-any.whl", hash = "sha256:2335065e6395b9e67ca716de5f7526736bfa6ceead690adf616d925bdc622b13"}, - {file = "Click-7.0.tar.gz", hash = "sha256:5b94b49521f6456670fdb30cd82a4eca9412788a93fa6dd6df72c94d5a8ff2d7"}, -] -colorama = [ - {file = "colorama-0.4.3-py2.py3-none-any.whl", hash = "sha256:7d73d2a99753107a36ac6b455ee49046802e59d9d076ef8e47b61499fa29afff"}, - {file = "colorama-0.4.3.tar.gz", hash = "sha256:e96da0d330793e2cb9485e9ddfd918d456036c7149416295932478192f4436a1"}, -] -fastapi = [ - {file = "fastapi-0.48.0-py3-none-any.whl", hash = "sha256:87d409d3ac3957713c016ba3b28fa5214e0903d4351cc3fe486b170d29e8aacd"}, - {file = "fastapi-0.48.0.tar.gz", hash = "sha256:2e00347f6a84291a5f04302733fbcf7c2ad9c674c0d0448cbee661db0e01ca16"}, -] -h11 = [ - {file = "h11-0.9.0-py2.py3-none-any.whl", hash = "sha256:4bc6d6a1238b7615b266ada57e0618568066f57dd6fa967d1290ec9309b2f2f1"}, - {file = "h11-0.9.0.tar.gz", hash = "sha256:33d4bca7be0fa039f4e84d50ab00531047e53d6ee8ffbc83501ea602c169cae1"}, -] -h2 = [ - {file = "h2-3.2.0-py2.py3-none-any.whl", hash = "sha256:61e0f6601fa709f35cdb730863b4e5ec7ad449792add80d1410d4174ed139af5"}, - {file = "h2-3.2.0.tar.gz", hash = "sha256:875f41ebd6f2c44781259005b157faed1a5031df3ae5aa7bcb4628a6c0782f14"}, -] -hpack = [ - {file = "hpack-3.0.0-py2.py3-none-any.whl", hash = "sha256:0edd79eda27a53ba5be2dfabf3b15780928a0dff6eb0c60a3d6767720e970c89"}, - {file = "hpack-3.0.0.tar.gz", hash = "sha256:8eec9c1f4bfae3408a3f30500261f7e6a65912dc138526ea054f9ad98892e9d2"}, -] -hstspreload = [ - {file = "hstspreload-2020.2.25.tar.gz", hash = "sha256:a1ba0c2730593a1922f93cd9c66ff620248090656102bf31e4559c01d7935e05"}, -] -httptools = [ - {file = "httptools-0.1.1-cp35-cp35m-macosx_10_13_x86_64.whl", hash = "sha256:a2719e1d7a84bb131c4f1e0cb79705034b48de6ae486eb5297a139d6a3296dce"}, - {file = "httptools-0.1.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:fa3cd71e31436911a44620473e873a256851e1f53dee56669dae403ba41756a4"}, - {file = "httptools-0.1.1-cp36-cp36m-macosx_10_13_x86_64.whl", hash = "sha256:86c6acd66765a934e8730bf0e9dfaac6fdcf2a4334212bd4a0a1c78f16475ca6"}, - {file = "httptools-0.1.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:bc3114b9edbca5a1eb7ae7db698c669eb53eb8afbbebdde116c174925260849c"}, - {file = "httptools-0.1.1-cp36-cp36m-win_amd64.whl", hash = "sha256:ac0aa11e99454b6a66989aa2d44bca41d4e0f968e395a0a8f164b401fefe359a"}, - {file = "httptools-0.1.1-cp37-cp37m-macosx_10_13_x86_64.whl", hash = "sha256:96da81e1992be8ac2fd5597bf0283d832287e20cb3cfde8996d2b00356d4e17f"}, - {file = "httptools-0.1.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:56b6393c6ac7abe632f2294da53f30d279130a92e8ae39d8d14ee2e1b05ad1f2"}, - {file = "httptools-0.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:96eb359252aeed57ea5c7b3d79839aaa0382c9d3149f7d24dd7172b1bcecb009"}, - {file = "httptools-0.1.1-cp38-cp38-macosx_10_13_x86_64.whl", hash = "sha256:fea04e126014169384dee76a153d4573d90d0cbd1d12185da089f73c78390437"}, - {file = "httptools-0.1.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:3592e854424ec94bd17dc3e0c96a64e459ec4147e6d53c0a42d0ebcef9cb9c5d"}, - {file = "httptools-0.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:0a4b1b2012b28e68306575ad14ad5e9120b34fccd02a81eb08838d7e3bbb48be"}, - {file = "httptools-0.1.1.tar.gz", hash = "sha256:41b573cf33f64a8f8f3400d0a7faf48e1888582b6f6e02b82b9bd4f0bf7497ce"}, -] -httpx = [ - {file = "httpx-0.11.1-py2.py3-none-any.whl", hash = "sha256:1d3893d3e4244c569764a6bae5c5a9fbbc4a6ec3825450b5696602af7a275576"}, - {file = "httpx-0.11.1.tar.gz", hash = "sha256:7d2bfb726eeed717953d15dddb22da9c2fcf48a4d70ba1456aa0a7faeda33cf7"}, -] -hyperframe = [ - {file = "hyperframe-5.2.0-py2.py3-none-any.whl", hash = "sha256:5187962cb16dcc078f23cb5a4b110098d546c3f41ff2d4038a9896893bbd0b40"}, - {file = "hyperframe-5.2.0.tar.gz", hash = "sha256:a9f5c17f2cc3c719b917c4f33ed1c61bd1f8dfac4b1bd23b7c80b3400971b41f"}, -] -idna = [ - {file = "idna-2.9-py2.py3-none-any.whl", hash = "sha256:a068a21ceac8a4d63dbfd964670474107f541babbd2250d61922f029858365fa"}, - {file = "idna-2.9.tar.gz", hash = "sha256:7588d1c14ae4c77d74036e8c22ff447b26d0fde8f007354fd48a7814db15b7cb"}, -] -isort = [ - {file = "isort-4.3.21-py2.py3-none-any.whl", hash = "sha256:6e811fcb295968434526407adb8796944f1988c5b65e8139058f2014cbe100fd"}, - {file = "isort-4.3.21.tar.gz", hash = "sha256:54da7e92468955c4fceacd0c86bd0ec997b0e1ee80d97f67c35a78b719dccab1"}, -] -loguru = [ - {file = "loguru-0.4.1-py3-none-any.whl", hash = "sha256:074b3caa6748452c1e4f2b302093c94b65d5a4c5a4d7743636b4121e06437b0e"}, - {file = "loguru-0.4.1.tar.gz", hash = "sha256:a6101fd435ac89ba5205a105a26a6ede9e4ddbb4408a6e167852efca47806d11"}, -] -pathspec = [ - {file = "pathspec-0.7.0-py2.py3-none-any.whl", hash = "sha256:163b0632d4e31cef212976cf57b43d9fd6b0bac6e67c26015d611a647d5e7424"}, - {file = "pathspec-0.7.0.tar.gz", hash = "sha256:562aa70af2e0d434367d9790ad37aed893de47f1693e4201fd1d3dca15d19b96"}, -] -pydantic = [ - {file = "pydantic-1.4-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:07911aab70f3bc52bb845ce1748569c5e70478ac977e106a150dd9d0465ebf04"}, - {file = "pydantic-1.4-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:012c422859bac2e03ab3151ea6624fecf0e249486be7eb8c6ee69c91740c6752"}, - {file = "pydantic-1.4-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:61d22d36808087d3184ed6ac0d91dd71c533b66addb02e4a9930e1e30833202f"}, - {file = "pydantic-1.4-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:f863456d3d4bf817f2e5248553dee3974c5dc796f48e6ddb599383570f4215ac"}, - {file = "pydantic-1.4-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:bbbed364376f4a0aebb9ea452ff7968b306499a9e74f4db69b28ff2cd4043a11"}, - {file = "pydantic-1.4-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e27559cedbd7f59d2375bfd6eea29a330ea1a5b0589c34d6b4e0d7bec6027bbf"}, - {file = "pydantic-1.4-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:50e4e948892a6815649ad5a9a9379ad1e5f090f17842ac206535dfaed75c6f2f"}, - {file = "pydantic-1.4-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:8848b4eb458469739126e4c1a202d723dd092e087f8dbe3104371335f87ba5df"}, - {file = "pydantic-1.4-cp38-cp38-manylinux1_i686.whl", hash = "sha256:831a0265a9e3933b3d0f04d1a81bba543bafbe4119c183ff2771871db70524ab"}, - {file = "pydantic-1.4-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:47b8db7024ba3d46c3d4768535e1cf87b6c8cf92ccd81e76f4e1cb8ee47688b3"}, - {file = "pydantic-1.4-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:51f11c8bbf794a68086540da099aae4a9107447c7a9d63151edbb7d50110cf21"}, - {file = "pydantic-1.4-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:6100d7862371115c40be55cc4b8d766a74b1d0dbaf99dbfe72bb4bac0faf89ed"}, - {file = "pydantic-1.4-py36.py37.py38-none-any.whl", hash = "sha256:72184c1421103cca128300120f8f1185fb42a9ea73a1c9845b1c53db8c026a7d"}, - {file = "pydantic-1.4.tar.gz", hash = "sha256:f17ec336e64d4583311249fb179528e9a2c27c8a2eaf590ec6ec2c6dece7cb3f"}, -] -pyflakes = [ - {file = "pyflakes-2.1.1-py2.py3-none-any.whl", hash = "sha256:17dbeb2e3f4d772725c777fabc446d5634d1038f234e77343108ce445ea69ce0"}, - {file = "pyflakes-2.1.1.tar.gz", hash = "sha256:d976835886f8c5b31d47970ed689944a0262b5f3afa00a5a7b4dc81e5449f8a2"}, -] -regex = [ - {file = "regex-2020.2.20-cp27-cp27m-win32.whl", hash = "sha256:99272d6b6a68c7ae4391908fc15f6b8c9a6c345a46b632d7fdb7ef6c883a2bbb"}, - {file = "regex-2020.2.20-cp27-cp27m-win_amd64.whl", hash = "sha256:974535648f31c2b712a6b2595969f8ab370834080e00ab24e5dbb9d19b8bfb74"}, - {file = "regex-2020.2.20-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:5de40649d4f88a15c9489ed37f88f053c15400257eeb18425ac7ed0a4e119400"}, - {file = "regex-2020.2.20-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:82469a0c1330a4beb3d42568f82dffa32226ced006e0b063719468dcd40ffdf0"}, - {file = "regex-2020.2.20-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:d58a4fa7910102500722defbde6e2816b0372a4fcc85c7e239323767c74f5cbc"}, - {file = "regex-2020.2.20-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:f1ac2dc65105a53c1c2d72b1d3e98c2464a133b4067a51a3d2477b28449709a0"}, - {file = "regex-2020.2.20-cp36-cp36m-win32.whl", hash = "sha256:8c2b7fa4d72781577ac45ab658da44c7518e6d96e2a50d04ecb0fd8f28b21d69"}, - {file = "regex-2020.2.20-cp36-cp36m-win_amd64.whl", hash = "sha256:269f0c5ff23639316b29f31df199f401e4cb87529eafff0c76828071635d417b"}, - {file = "regex-2020.2.20-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:bed7986547ce54d230fd8721aba6fd19459cdc6d315497b98686d0416efaff4e"}, - {file = "regex-2020.2.20-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:046e83a8b160aff37e7034139a336b660b01dbfe58706f9d73f5cdc6b3460242"}, - {file = "regex-2020.2.20-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:b33ebcd0222c1d77e61dbcd04a9fd139359bded86803063d3d2d197b796c63ce"}, - {file = "regex-2020.2.20-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:bba52d72e16a554d1894a0cc74041da50eea99a8483e591a9edf1025a66843ab"}, - {file = "regex-2020.2.20-cp37-cp37m-win32.whl", hash = "sha256:01b2d70cbaed11f72e57c1cfbaca71b02e3b98f739ce33f5f26f71859ad90431"}, - {file = "regex-2020.2.20-cp37-cp37m-win_amd64.whl", hash = "sha256:113309e819634f499d0006f6200700c8209a2a8bf6bd1bdc863a4d9d6776a5d1"}, - {file = "regex-2020.2.20-cp38-cp38-manylinux1_i686.whl", hash = "sha256:25f4ce26b68425b80a233ce7b6218743c71cf7297dbe02feab1d711a2bf90045"}, - {file = "regex-2020.2.20-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:9b64a4cc825ec4df262050c17e18f60252cdd94742b4ba1286bcfe481f1c0f26"}, - {file = "regex-2020.2.20-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:9ff16d994309b26a1cdf666a6309c1ef51ad4f72f99d3392bcd7b7139577a1f2"}, - {file = "regex-2020.2.20-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:c7f58a0e0e13fb44623b65b01052dae8e820ed9b8b654bb6296bc9c41f571b70"}, - {file = "regex-2020.2.20-cp38-cp38-win32.whl", hash = "sha256:200539b5124bc4721247a823a47d116a7a23e62cc6695744e3eb5454a8888e6d"}, - {file = "regex-2020.2.20-cp38-cp38-win_amd64.whl", hash = "sha256:7f78f963e62a61e294adb6ff5db901b629ef78cb2a1cfce3cf4eeba80c1c67aa"}, - {file = "regex-2020.2.20.tar.gz", hash = "sha256:9e9624440d754733eddbcd4614378c18713d2d9d0dc647cf9c72f64e39671be5"}, -] -rfc3986 = [ - {file = "rfc3986-1.3.2-py2.py3-none-any.whl", hash = "sha256:df4eba676077cefb86450c8f60121b9ae04b94f65f85b69f3f731af0516b7b18"}, - {file = "rfc3986-1.3.2.tar.gz", hash = "sha256:0344d0bd428126ce554e7ca2b61787b6a28d2bbd19fc70ed2dd85efe31176405"}, -] -sniffio = [ - {file = "sniffio-1.1.0-py3-none-any.whl", hash = "sha256:20ed6d5b46f8ae136d00b9dcb807615d83ed82ceea6b2058cecb696765246da5"}, - {file = "sniffio-1.1.0.tar.gz", hash = "sha256:8e3810100f69fe0edd463d02ad407112542a11ffdc29f67db2bf3771afb87a21"}, -] -starlette = [ - {file = "starlette-0.12.9.tar.gz", hash = "sha256:c2ac9a42e0e0328ad20fe444115ac5e3760c1ee2ac1ff8cdb5ec915c4a453411"}, -] -toml = [ - {file = "toml-0.10.0-py2.7.egg", hash = "sha256:f1db651f9657708513243e61e6cc67d101a39bad662eaa9b5546f789338e07a3"}, - {file = "toml-0.10.0-py2.py3-none-any.whl", hash = "sha256:235682dd292d5899d361a811df37e04a8828a5b1da3115886b73cf81ebc9100e"}, - {file = "toml-0.10.0.tar.gz", hash = "sha256:229f81c57791a41d65e399fc06bf0848bab550a9dfd5ed66df18ce5f05e73d5c"}, -] -typed-ast = [ - {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:73d785a950fc82dd2a25897d525d003f6378d1cb23ab305578394694202a58c3"}, - {file = "typed_ast-1.4.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:aaee9905aee35ba5905cfb3c62f3e83b3bec7b39413f0a7f19be4e547ea01ebb"}, - {file = "typed_ast-1.4.1-cp35-cp35m-win32.whl", hash = "sha256:0c2c07682d61a629b68433afb159376e24e5b2fd4641d35424e462169c0a7919"}, - {file = "typed_ast-1.4.1-cp35-cp35m-win_amd64.whl", hash = "sha256:4083861b0aa07990b619bd7ddc365eb7fa4b817e99cf5f8d9cf21a42780f6e01"}, - {file = "typed_ast-1.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:269151951236b0f9a6f04015a9004084a5ab0d5f19b57de779f908621e7d8b75"}, - {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:24995c843eb0ad11a4527b026b4dde3da70e1f2d8806c99b7b4a7cf491612652"}, - {file = "typed_ast-1.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:fe460b922ec15dd205595c9b5b99e2f056fd98ae8f9f56b888e7a17dc2b757e7"}, - {file = "typed_ast-1.4.1-cp36-cp36m-win32.whl", hash = "sha256:4e3e5da80ccbebfff202a67bf900d081906c358ccc3d5e3c8aea42fdfdfd51c1"}, - {file = "typed_ast-1.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:249862707802d40f7f29f6e1aad8d84b5aa9e44552d2cc17384b209f091276aa"}, - {file = "typed_ast-1.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:8ce678dbaf790dbdb3eba24056d5364fb45944f33553dd5869b7580cdbb83614"}, - {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:c9e348e02e4d2b4a8b2eedb48210430658df6951fa484e59de33ff773fbd4b41"}, - {file = "typed_ast-1.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:bcd3b13b56ea479b3650b82cabd6b5343a625b0ced5429e4ccad28a8973f301b"}, - {file = "typed_ast-1.4.1-cp37-cp37m-win32.whl", hash = "sha256:d5d33e9e7af3b34a40dc05f498939f0ebf187f07c385fd58d591c533ad8562fe"}, - {file = "typed_ast-1.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:0666aa36131496aed8f7be0410ff974562ab7eeac11ef351def9ea6fa28f6355"}, - {file = "typed_ast-1.4.1-cp38-cp38-macosx_10_15_x86_64.whl", hash = "sha256:d205b1b46085271b4e15f670058ce182bd1199e56b317bf2ec004b6a44f911f6"}, - {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:6daac9731f172c2a22ade6ed0c00197ee7cc1221aa84cfdf9c31defeb059a907"}, - {file = "typed_ast-1.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:498b0f36cc7054c1fead3d7fc59d2150f4d5c6c56ba7fb150c013fbc683a8d2d"}, - {file = "typed_ast-1.4.1-cp38-cp38-win32.whl", hash = "sha256:715ff2f2df46121071622063fc7543d9b1fd19ebfc4f5c8895af64a77a8c852c"}, - {file = "typed_ast-1.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:fc0fea399acb12edbf8a628ba8d2312f583bdbdb3335635db062fa98cf71fca4"}, - {file = "typed_ast-1.4.1-cp39-cp39-macosx_10_15_x86_64.whl", hash = "sha256:d43943ef777f9a1c42bf4e552ba23ac77a6351de620aa9acf64ad54933ad4d34"}, - {file = "typed_ast-1.4.1.tar.gz", hash = "sha256:8c8aaad94455178e3187ab22c8b01a3837f8ee50e09cf31f1ba129eb293ec30b"}, -] -urllib3 = [ - {file = "urllib3-1.25.8-py2.py3-none-any.whl", hash = "sha256:2f3db8b19923a873b3e5256dc9c2dedfa883e33d87c690d9c7913e1f40673cdc"}, - {file = "urllib3-1.25.8.tar.gz", hash = "sha256:87716c2d2a7121198ebcb7ce7cccf6ce5e9ba539041cfbaeecfb641dc0bf6acc"}, -] -uvicorn = [ - {file = "uvicorn-0.11.3-py3-none-any.whl", hash = "sha256:0f58170165c4495f563d8224b2f415a0829af0412baa034d6f777904613087fd"}, - {file = "uvicorn-0.11.3.tar.gz", hash = "sha256:6fdaf8e53bf1b2ddf0fe9ed06079b5348d7d1d87b3365fe2549e6de0d49e631c"}, -] -uvloop = [ - {file = "uvloop-0.14.0-cp35-cp35m-macosx_10_11_x86_64.whl", hash = "sha256:08b109f0213af392150e2fe6f81d33261bb5ce968a288eb698aad4f46eb711bd"}, - {file = "uvloop-0.14.0-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:4544dcf77d74f3a84f03dd6278174575c44c67d7165d4c42c71db3fdc3860726"}, - {file = "uvloop-0.14.0-cp36-cp36m-macosx_10_11_x86_64.whl", hash = "sha256:b4f591aa4b3fa7f32fb51e2ee9fea1b495eb75b0b3c8d0ca52514ad675ae63f7"}, - {file = "uvloop-0.14.0-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:f07909cd9fc08c52d294b1570bba92186181ca01fe3dc9ffba68955273dd7362"}, - {file = "uvloop-0.14.0-cp37-cp37m-macosx_10_11_x86_64.whl", hash = "sha256:afd5513c0ae414ec71d24f6f123614a80f3d27ca655a4fcf6cabe50994cc1891"}, - {file = "uvloop-0.14.0-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:e7514d7a48c063226b7d06617cbb12a14278d4323a065a8d46a7962686ce2e95"}, - {file = "uvloop-0.14.0-cp38-cp38-macosx_10_11_x86_64.whl", hash = "sha256:bcac356d62edd330080aed082e78d4b580ff260a677508718f88016333e2c9c5"}, - {file = "uvloop-0.14.0-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:4315d2ec3ca393dd5bc0b0089d23101276778c304d42faff5dc4579cb6caef09"}, - {file = "uvloop-0.14.0.tar.gz", hash = "sha256:123ac9c0c7dd71464f58f1b4ee0bbd81285d96cdda8bc3519281b8973e3a461e"}, -] -websockets = [ - {file = "websockets-8.1-cp36-cp36m-macosx_10_6_intel.whl", hash = "sha256:3762791ab8b38948f0c4d281c8b2ddfa99b7e510e46bd8dfa942a5fff621068c"}, - {file = "websockets-8.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:3db87421956f1b0779a7564915875ba774295cc86e81bc671631379371af1170"}, - {file = "websockets-8.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4f9f7d28ce1d8f1295717c2c25b732c2bc0645db3215cf757551c392177d7cb8"}, - {file = "websockets-8.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:295359a2cc78736737dd88c343cd0747546b2174b5e1adc223824bcaf3e164cb"}, - {file = "websockets-8.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:1d3f1bf059d04a4e0eb4985a887d49195e15ebabc42364f4eb564b1d065793f5"}, - {file = "websockets-8.1-cp36-cp36m-win32.whl", hash = "sha256:2db62a9142e88535038a6bcfea70ef9447696ea77891aebb730a333a51ed559a"}, - {file = "websockets-8.1-cp36-cp36m-win_amd64.whl", hash = "sha256:0e4fb4de42701340bd2353bb2eee45314651caa6ccee80dbd5f5d5978888fed5"}, - {file = "websockets-8.1-cp37-cp37m-macosx_10_6_intel.whl", hash = "sha256:9b248ba3dd8a03b1a10b19efe7d4f7fa41d158fdaa95e2cf65af5a7b95a4f989"}, - {file = "websockets-8.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ce85b06a10fc65e6143518b96d3dca27b081a740bae261c2fb20375801a9d56d"}, - {file = "websockets-8.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:965889d9f0e2a75edd81a07592d0ced54daa5b0785f57dc429c378edbcffe779"}, - {file = "websockets-8.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:751a556205d8245ff94aeef23546a1113b1dd4f6e4d102ded66c39b99c2ce6c8"}, - {file = "websockets-8.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:3ef56fcc7b1ff90de46ccd5a687bbd13a3180132268c4254fc0fa44ecf4fc422"}, - {file = "websockets-8.1-cp37-cp37m-win32.whl", hash = "sha256:7ff46d441db78241f4c6c27b3868c9ae71473fe03341340d2dfdbe8d79310acc"}, - {file = "websockets-8.1-cp37-cp37m-win_amd64.whl", hash = "sha256:20891f0dddade307ffddf593c733a3fdb6b83e6f9eef85908113e628fa5a8308"}, - {file = "websockets-8.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:c1ec8db4fac31850286b7cd3b9c0e1b944204668b8eb721674916d4e28744092"}, - {file = "websockets-8.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:5c01fd846263a75bc8a2b9542606927cfad57e7282965d96b93c387622487485"}, - {file = "websockets-8.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:9bef37ee224e104a413f0780e29adb3e514a5b698aabe0d969a6ba426b8435d1"}, - {file = "websockets-8.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:d705f8aeecdf3262379644e4b55107a3b55860eb812b673b28d0fbc347a60c55"}, - {file = "websockets-8.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:c8a116feafdb1f84607cb3b14aa1418424ae71fee131642fc568d21423b51824"}, - {file = "websockets-8.1-cp38-cp38-win32.whl", hash = "sha256:e898a0863421650f0bebac8ba40840fc02258ef4714cb7e1fd76b6a6354bda36"}, - {file = "websockets-8.1-cp38-cp38-win_amd64.whl", hash = "sha256:f8a7bff6e8664afc4e6c28b983845c5bc14965030e3fb98789734d416af77c4b"}, - {file = "websockets-8.1.tar.gz", hash = "sha256:5c65d2da8c6bce0fca2528f69f44b2f977e06954c8512a952222cea50dad430f"}, -] -win32-setctime = [ - {file = "win32_setctime-1.0.1-py3-none-any.whl", hash = "sha256:568fd636c68350bcc54755213fe01966fe0a6c90b386c0776425944a0382abef"}, - {file = "win32_setctime-1.0.1.tar.gz", hash = "sha256:b47e5023ec7f0b4962950902b15bc56464a380d869f59d27dbf9ab423b23e8f9"}, -] diff --git a/examples/fsm/pyproject.toml b/examples/fsm/pyproject.toml deleted file mode 100644 index ee91f8b6..00000000 --- a/examples/fsm/pyproject.toml +++ /dev/null @@ -1,21 +0,0 @@ -[tool.poetry] -name = "pybotx-fsm-example" -version = "0.0.0" -description = "An example of bot with FSM behaviour" -authors = ["Nik Sidnev "] -license = "MIT" - -[tool.poetry.dependencies] -python = "^3.8" -botx = "^0.13.0" -fastapi = "^0.48.0" -uvicorn = "^0.11.3" - -[tool.poetry.dev-dependencies] -black = "^19.10b0" -isort = "^4.3" -autoflake = "^1.3" - -[build-system] -requires = ["poetry>=1.0"] -build-backend = "poetry.masonry.api" diff --git a/examples/fsm/scripts/format b/examples/fsm/scripts/format deleted file mode 100755 index 9fa80f09..00000000 --- a/examples/fsm/scripts/format +++ /dev/null @@ -1,12 +0,0 @@ -#!/usr/bin/env bash - -set -e - -# Sort imports one per line, so autoflake can remove unused imports -isort --recursive --force-single-line-imports bot - -autoflake --recursive --remove-all-unused-imports --remove-unused-variables --in-place bot -black bot -isort --recursive bot - -isort --recursive --thirdparty=botx bot diff --git a/mkdocs.yml b/mkdocs.yml index 10174435..b22de101 100644 --- a/mkdocs.yml +++ b/mkdocs.yml @@ -1,9 +1,9 @@ -site_name: pybotx -site_url: !!python/object/apply:os.getenv ["SITE_URL", "https://expressapp.github.io/pybotx"] +site_name: pybotx-next +site_url: !!python/object/apply:os.getenv ["SITE_URL", "https://expressapp.github.io/pybotx-next"] site_description: A little python library for building bots for Express theme: - name: 'material' + name: "material" repo_name: ExpressApp/pybotx repo_url: https://github.com/ExpressApp/pybotx @@ -11,41 +11,20 @@ edit_uri: '' nav: - Introduction: 'index.md' - - Tutorial - User Guide: - - First Steps: 'development/first-steps.md' - - Sending Messages And Files: 'development/sending-data.md' - - Handlers Collecting: 'development/collector.md' - - Handling Errors: 'development/handling-errors.md' - - Dependencies Injection: 'development/dependencies-injection.md' - - Logging: 'development/logging.md' - - Testing: 'development/tests.md' - - API Reference: - - Bots: 'reference/bots.md' - - Collecting: 'reference/collecting.md' - - Middlewares: - - Base: 'reference/middlewares/base.md' - - Next Step: 'reference/middlewares/ns.md' - - Authorization: 'reference/middlewares/authorization.md' - - Models: 'reference/models.md' - - Clients: - - Methods: 'reference/clients/methods.md' - - Async Client: 'reference/clients/async-client.md' - - Synchronous Client: 'reference/clients/sync-client.md' - - Testing: - Client: 'reference/testing/test-client.md' - Message Builder: 'reference/testing/message-builder.md' - - Exceptions: 'reference/exceptions.md' + - BotX API: 'botx_api.md' - - Changelog: 'changelog.md' +plugins: + - search markdown_extensions: - - markdown.extensions.codehilite: - guess_lang: false - - markdown_include.include: - base_path: docs - admonition - - codehilite + - pymdownx.snippets + - pymdownx.highlight: + anchor_linenums: true + linenums: true + - pymdownx.superfences + - codehilite: + css_class: highlight -plugins: - - search - - mkdocstrings +extra_css: + - css/custom.css diff --git a/noxfile.py b/noxfile.py deleted file mode 100644 index ddad127a..00000000 --- a/noxfile.py +++ /dev/null @@ -1,93 +0,0 @@ -"""Run common tasks using nox.""" -import pathlib - -import nox -from nox.sessions import Session - -TARGETS = ("botx", "tests") - - -def _process_add_single_comma_path(session: Session, path: pathlib.Path) -> None: - if path.is_dir(): - for new_path in path.iterdir(): - _process_add_single_comma_path(session, new_path) - - return - - if path.suffix not in {".py", ".pyi"}: - return - - session.run( - "add-trailing-comma", "--py36-plus", "--exit-zero-even-if-changed", str(path), - ) - - -def _process_add_single_comma(session: Session, *paths: str) -> None: - for target in paths: - path = pathlib.Path(target) - _process_add_single_comma_path(session, path) - - -@nox.session(python=False, name="format") -def run_formatters(session: Session) -> None: - """Run all project formatters. - - Formatters to run: - 1. isort with autoflake to remove all unused imports. - 2. black for sinle style in all project. - 3. add-trailing-comma to adding or removing comma from line. - 4. isort for properly imports sorting. - """ - # we need to run isort here, since autoflake is unable to understand unused imports - # when they are multiline. - # see https://github.com/myint/autoflake/issues/8 - session.run("isort", "--recursive", "--force-single-line-imports", *TARGETS) - session.run( - "autoflake", - "--recursive", - "--remove-all-unused-imports", - "--remove-unused-variables", - "--in-place", - *TARGETS, - ) - session.run("black", *TARGETS) - _process_add_single_comma(session, *TARGETS) - session.run("isort", "--recursive", *TARGETS) - - -@nox.session(python=False) -def lint(session: Session) -> None: - """Run all project linters. - - Linters to run: - 1. black for code format style. - 2. mypy for type checking. - 3. flake8 for common python code style issues. - """ - session.run("black", "--check", "--diff", *TARGETS) - session.run("mypy", *TARGETS) - session.run("flake8", *TARGETS) - - -@nox.session(python=False) -def test(session: Session) -> None: - """Run pytest.""" - session.run("pytest", "--cov-config=setup.cfg") - - -@nox.session(python=False) -def publish(session: Session) -> None: - """Publish library on PyPI.""" - session.run("poetry", "publish", "--build") - - -@nox.session(python=False, name="build-docs") -def build_docs(session: Session) -> None: - """Build MkDocs pages.""" - session.run("mkdocs", "build") - - -@nox.session(python=False, name="serve-docs") -def serve_docs(session: Session) -> None: - """Serve MkDocs pages.""" - session.run("mkdocs", "serve", "--dev-addr", "0.0.0.0:8008") diff --git a/poetry.lock b/poetry.lock index 2e0c9328..3d8ec0d3 100644 --- a/poetry.lock +++ b/poetry.lock @@ -1,6 +1,6 @@ [[package]] name = "add-trailing-comma" -version = "2.1.0" +version = "2.2.1" description = "Automatically add trailing commas to calls and literals" category = "dev" optional = false @@ -11,15 +11,15 @@ tokenize-rt = ">=3.0.1" [[package]] name = "aiofiles" -version = "0.7.0" +version = "0.8.0" description = "File support for asyncio." category = "main" -optional = true +optional = false python-versions = ">=3.6,<4.0" [[package]] name = "anyio" -version = "3.3.1" +version = "3.5.0" description = "High level compatibility layer for multiple asynchronous event loop implementations" category = "main" optional = false @@ -28,34 +28,22 @@ python-versions = ">=3.6.2" [package.dependencies] idna = ">=2.8" sniffio = ">=1.1" -typing-extensions = {version = "*", markers = "python_version < \"3.8\""} [package.extras] -doc = ["sphinx-rtd-theme", "sphinx-autodoc-typehints (>=1.2.0)"] -test = ["coverage[toml] (>=4.5)", "hypothesis (>=4.0)", "pytest (>=6.0)", "pytest-mock (>=3.6.1)", "trustme", "uvloop (<0.15)", "mock (>=4)", "uvloop (>=0.15)"] +doc = ["packaging", "sphinx-rtd-theme", "sphinx-autodoc-typehints (>=1.2.0)"] +test = ["coverage[toml] (>=4.5)", "hypothesis (>=4.0)", "pytest (>=6.0)", "pytest-mock (>=3.6.1)", "trustme", "contextlib2", "uvloop (<0.15)", "mock (>=4)", "uvloop (>=0.15)"] trio = ["trio (>=0.16)"] [[package]] -name = "appdirs" -version = "1.4.4" -description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." -category = "dev" -optional = false -python-versions = "*" - -[[package]] -name = "argcomplete" -version = "1.12.3" -description = "Bash tab completion for argparse" +name = "asgiref" +version = "3.5.0" +description = "ASGI specs, helper code, and adapters" category = "dev" optional = false -python-versions = "*" - -[package.dependencies] -importlib-metadata = {version = ">=0.23,<5", markers = "python_version == \"3.7\""} +python-versions = ">=3.7" [package.extras] -test = ["coverage", "flake8", "pexpect", "wheel"] +tests = ["pytest", "pytest-asyncio", "mypy (>=0.800)"] [[package]] name = "astor" @@ -75,17 +63,17 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" [[package]] name = "attrs" -version = "21.2.0" +version = "21.4.0" description = "Classes Without Boilerplate" category = "dev" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" [package.extras] -dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "furo", "sphinx", "sphinx-notfound-page", "pre-commit"] +dev = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "furo", "sphinx", "sphinx-notfound-page", "pre-commit", "cloudpickle"] docs = ["furo", "sphinx", "zope.interface", "sphinx-notfound-page"] -tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface"] -tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins"] +tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface", "cloudpickle"] +tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "cloudpickle"] [[package]] name = "autoflake" @@ -98,69 +86,54 @@ python-versions = "*" [package.dependencies] pyflakes = ">=1.1.0" -[[package]] -name = "backports.entry-points-selectable" -version = "1.1.0" -description = "Compatibility shim providing selectable entry points for older implementations" -category = "dev" -optional = false -python-versions = ">=2.7" - -[package.dependencies] -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} - -[package.extras] -docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] -testing = ["pytest (>=4.6)", "pytest-flake8", "pytest-cov", "pytest-black (>=0.3.7)", "pytest-mypy", "pytest-checkdocs (>=2.4)", "pytest-enabler (>=1.0.1)"] - [[package]] name = "bandit" -version = "1.7.0" +version = "1.7.2" description = "Security oriented static analyser for python code." category = "dev" optional = false -python-versions = ">=3.5" +python-versions = ">=3.7" [package.dependencies] colorama = {version = ">=0.3.9", markers = "platform_system == \"Windows\""} GitPython = ">=1.0.1" PyYAML = ">=5.3.1" -six = ">=1.10.0" stevedore = ">=1.20.0" -[[package]] -name = "base64io" -version = "1.0.3" -description = "" -category = "main" -optional = false -python-versions = "*" +[package.extras] +test = ["beautifulsoup4 (>=4.8.0)", "coverage (>=4.5.4)", "fixtures (>=3.0.0)", "flake8 (>=4.0.0)", "pylint (==1.9.4)", "stestr (>=2.5.0)", "testscenarios (>=0.5.0)", "testtools (>=2.3.0)", "toml"] +toml = ["toml"] +yaml = ["pyyaml"] [[package]] name = "black" -version = "20.8b1" +version = "21.12b0" description = "The uncompromising code formatter." category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.6.2" [package.dependencies] -appdirs = "*" click = ">=7.1.2" mypy-extensions = ">=0.4.3" -pathspec = ">=0.6,<1" -regex = ">=2020.1.8" -toml = ">=0.10.1" -typed-ast = ">=1.4.0" -typing-extensions = ">=3.7.4" +pathspec = ">=0.9.0,<1" +platformdirs = ">=2" +tomli = ">=0.2.6,<2.0.0" +typing-extensions = [ + {version = ">=3.10.0.0", markers = "python_version < \"3.10\""}, + {version = "!=3.10.0.1", markers = "python_version >= \"3.10\""}, +] [package.extras] colorama = ["colorama (>=0.4.3)"] -d = ["aiohttp (>=3.3.2)", "aiohttp-cors"] +d = ["aiohttp (>=3.7.4)"] +jupyter = ["ipython (>=7.8.0)", "tokenize-rt (>=3.2.0)"] +python2 = ["typed-ast (>=1.4.3)"] +uvloop = ["uvloop (>=0.15.2)"] [[package]] name = "certifi" -version = "2021.5.30" +version = "2021.10.8" description = "Python package for providing Mozilla's CA Bundle." category = "main" optional = false @@ -168,7 +141,7 @@ python-versions = "*" [[package]] name = "charset-normalizer" -version = "2.0.4" +version = "2.0.12" description = "The Real First Universal Charset Detector. Open, modern and actively maintained alternative to Chardet." category = "main" optional = false @@ -179,7 +152,7 @@ unicode_backport = ["unicodedata2"] [[package]] name = "click" -version = "8.0.1" +version = "8.0.4" description = "Composable command line interface toolkit" category = "dev" optional = false @@ -187,7 +160,6 @@ python-versions = ">=3.6" [package.dependencies] colorama = {version = "*", markers = "platform_system == \"Windows\""} -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} [[package]] name = "colorama" @@ -197,59 +169,31 @@ category = "main" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" -[[package]] -name = "colorlog" -version = "4.8.0" -description = "Log formatting with colors!" -category = "dev" -optional = false -python-versions = "*" - -[package.dependencies] -colorama = {version = "*", markers = "sys_platform == \"win32\""} - [[package]] name = "coverage" -version = "5.5" +version = "6.3.2" description = "Code coverage measurement for Python" category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" - -[package.extras] -toml = ["toml"] - -[[package]] -name = "coverage-conditional-plugin" -version = "0.4.0" -description = "Conditional coverage based on any rules you define!" -category = "dev" -optional = false -python-versions = ">=3.6,<4.0" +python-versions = ">=3.7" [package.dependencies] -coverage = ">=5.0,<6.0" -packaging = ">=20.4" +tomli = {version = "*", optional = true, markers = "extra == \"toml\""} + +[package.extras] +toml = ["tomli"] [[package]] name = "darglint" -version = "1.8.0" +version = "1.8.1" description = "A utility for ensuring Google-style docstrings stay up to date with the source code." category = "dev" optional = false python-versions = ">=3.6,<4.0" -[[package]] -name = "distlib" -version = "0.3.2" -description = "Distribution utilities" -category = "dev" -optional = false -python-versions = "*" - [[package]] name = "docutils" -version = "0.17.1" +version = "0.18.1" description = "Docutils -- Python Documentation Utilities" category = "dev" optional = false @@ -264,26 +208,35 @@ optional = false python-versions = "*" [[package]] -name = "filelock" -version = "3.0.12" -description = "A platform independent file lock." +name = "fastapi" +version = "0.73.0" +description = "FastAPI framework, high performance, easy to learn, fast to code, ready for production" category = "dev" optional = false -python-versions = "*" +python-versions = ">=3.6.1" + +[package.dependencies] +pydantic = ">=1.6.2,<1.7 || >1.7,<1.7.1 || >1.7.1,<1.7.2 || >1.7.2,<1.7.3 || >1.7.3,<1.8 || >1.8,<1.8.1 || >1.8.1,<2.0.0" +starlette = "0.17.1" + +[package.extras] +all = ["requests (>=2.24.0,<3.0.0)", "jinja2 (>=2.11.2,<4.0.0)", "python-multipart (>=0.0.5,<0.0.6)", "itsdangerous (>=1.1.0,<3.0.0)", "pyyaml (>=5.3.1,<6.0.0)", "ujson (>=4.0.1,<5.0.0)", "orjson (>=3.2.1,<4.0.0)", "email_validator (>=1.1.1,<2.0.0)", "uvicorn[standard] (>=0.12.0,<0.16.0)"] +dev = ["python-jose[cryptography] (>=3.3.0,<4.0.0)", "passlib[bcrypt] (>=1.7.2,<2.0.0)", "autoflake (>=1.4.0,<2.0.0)", "flake8 (>=3.8.3,<4.0.0)", "uvicorn[standard] (>=0.12.0,<0.16.0)"] +doc = ["mkdocs (>=1.1.2,<2.0.0)", "mkdocs-material (>=8.1.4,<9.0.0)", "mdx-include (>=1.4.1,<2.0.0)", "mkdocs-markdownextradata-plugin (>=0.1.7,<0.3.0)", "typer-cli (>=0.0.12,<0.0.13)", "pyyaml (>=5.3.1,<6.0.0)"] +test = ["pytest (>=6.2.4,<7.0.0)", "pytest-cov (>=2.12.0,<4.0.0)", "mypy (==0.910)", "flake8 (>=3.8.3,<4.0.0)", "black (==21.9b0)", "isort (>=5.0.6,<6.0.0)", "requests (>=2.24.0,<3.0.0)", "httpx (>=0.14.0,<0.19.0)", "email_validator (>=1.1.1,<2.0.0)", "sqlalchemy (>=1.3.18,<1.5.0)", "peewee (>=3.13.3,<4.0.0)", "databases[sqlite] (>=0.3.2,<0.6.0)", "orjson (>=3.2.1,<4.0.0)", "ujson (>=4.0.1,<5.0.0)", "python-multipart (>=0.0.5,<0.0.6)", "flask (>=1.1.2,<3.0.0)", "anyio[trio] (>=3.2.1,<4.0.0)", "types-ujson (==0.1.1)", "types-orjson (==3.6.0)", "types-dataclasses (==0.1.7)"] [[package]] name = "flake8" -version = "3.9.2" +version = "4.0.1" description = "the modular source code checker: pep8 pyflakes and co" category = "dev" optional = false -python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" +python-versions = ">=3.6" [package.dependencies] -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} mccabe = ">=0.6.0,<0.7.0" -pycodestyle = ">=2.7.0,<2.8.0" -pyflakes = ">=2.3.0,<2.4.0" +pycodestyle = ">=2.8.0,<2.9.0" +pyflakes = ">=2.4.0,<2.5.0" [[package]] name = "flake8-bandit" @@ -301,18 +254,18 @@ pycodestyle = "*" [[package]] name = "flake8-broken-line" -version = "0.3.0" +version = "0.4.0" description = "Flake8 plugin to forbid backslashes for line breaks" category = "dev" optional = false python-versions = ">=3.6,<4.0" [package.dependencies] -flake8 = ">=3.5,<4.0" +flake8 = ">=3.5,<5" [[package]] name = "flake8-bugbear" -version = "21.9.1" +version = "21.11.29" description = "A plugin for flake8 finding likely bugs and design problems in your program. Contains warnings that don't belong in pyflakes and pycodestyle." category = "dev" optional = false @@ -323,30 +276,29 @@ attrs = ">=19.2.0" flake8 = ">=3.0.0" [package.extras] -dev = ["coverage", "black", "hypothesis", "hypothesmith"] +dev = ["coverage", "hypothesis", "hypothesmith (>=0.2)", "pre-commit"] [[package]] name = "flake8-commas" -version = "2.0.0" +version = "2.1.0" description = "Flake8 lint for trailing commas." category = "dev" optional = false python-versions = "*" [package.dependencies] -flake8 = ">=2,<4.0.0" +flake8 = ">=2" [[package]] name = "flake8-comprehensions" -version = "3.6.1" +version = "3.8.0" description = "A flake8 plugin to help you write better list/set/dict comprehensions." category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [package.dependencies] -flake8 = ">=3.0,<3.2.0 || >3.2.0,<4" -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} +flake8 = ">=3.0,<3.2.0 || >3.2.0" [[package]] name = "flake8-debugger" @@ -375,7 +327,7 @@ pydocstyle = ">=2.1" [[package]] name = "flake8-eradicate" -version = "1.1.0" +version = "1.2.0" description = "Flake8 plugin to find commented out code" category = "dev" optional = false @@ -384,31 +336,23 @@ python-versions = ">=3.6,<4.0" [package.dependencies] attrs = "*" eradicate = ">=2.0,<3.0" -flake8 = ">=3.5,<4.0" +flake8 = ">=3.5,<5" [[package]] name = "flake8-isort" -version = "4.0.0" +version = "4.1.1" description = "flake8 plugin that integrates isort ." category = "dev" optional = false python-versions = "*" [package.dependencies] -flake8 = ">=3.2.1,<4" +flake8 = ">=3.2.1,<5" isort = ">=4.3.5,<6" testfixtures = ">=6.8.0,<7" [package.extras] -test = ["pytest (>=4.0.2,<6)", "toml"] - -[[package]] -name = "flake8-plugin-utils" -version = "1.3.2" -description = "The package provides base classes and utils for flake8 plugin writing" -category = "dev" -optional = false -python-versions = ">=3.6,<4.0" +test = ["pytest-cov"] [[package]] name = "flake8-polyfill" @@ -421,20 +365,9 @@ python-versions = "*" [package.dependencies] flake8 = "*" -[[package]] -name = "flake8-pytest-style" -version = "1.5.0" -description = "A flake8 plugin checking common style issues or inconsistencies with pytest-based tests." -category = "dev" -optional = false -python-versions = ">=3.6,<4.0" - -[package.dependencies] -flake8-plugin-utils = ">=1.3.2,<2.0.0" - [[package]] name = "flake8-quotes" -version = "3.3.0" +version = "3.3.1" description = "Flake8 lint for quotes." category = "dev" optional = false @@ -445,11 +378,11 @@ flake8 = "*" [[package]] name = "flake8-rst-docstrings" -version = "0.2.3" +version = "0.2.5" description = "Python docstring reStructuredText (RST) validator" category = "dev" optional = false -python-versions = ">=3.3" +python-versions = ">=3.6" [package.dependencies] flake8 = ">=3.0.0" @@ -469,7 +402,7 @@ flake8 = "*" [[package]] name = "ghp-import" -version = "2.0.1" +version = "2.0.2" description = "Copy your docs directly to the gh-pages branch." category = "dev" optional = false @@ -479,22 +412,22 @@ python-versions = "*" python-dateutil = ">=2.8.1" [package.extras] -dev = ["twine", "markdown", "flake8"] +dev = ["twine", "markdown", "flake8", "wheel"] [[package]] name = "gitdb" -version = "4.0.7" +version = "4.0.9" description = "Git Object Database" category = "dev" optional = false -python-versions = ">=3.4" +python-versions = ">=3.6" [package.dependencies] -smmap = ">=3.0.1,<5" +smmap = ">=3.0.1,<6" [[package]] name = "gitpython" -version = "3.1.23" +version = "3.1.27" description = "GitPython is a python library used to interact with Git repositories" category = "dev" optional = false @@ -502,7 +435,6 @@ python-versions = ">=3.7" [package.dependencies] gitdb = ">=4.0.1,<5" -typing-extensions = {version = ">=3.7.4.3", markers = "python_version < \"3.10\""} [[package]] name = "h11" @@ -514,7 +446,7 @@ python-versions = ">=3.6" [[package]] name = "httpcore" -version = "0.13.7" +version = "0.14.7" description = "A minimal low-level HTTP client." category = "main" optional = false @@ -522,15 +454,17 @@ python-versions = ">=3.6" [package.dependencies] anyio = ">=3.0.0,<4.0.0" +certifi = "*" h11 = ">=0.11,<0.13" sniffio = ">=1.0.0,<2.0.0" [package.extras] http2 = ["h2 (>=3,<5)"] +socks = ["socksio (>=1.0.0,<2.0.0)"] [[package]] name = "httpx" -version = "0.19.0" +version = "0.21.3" description = "The next generation HTTP client." category = "main" optional = false @@ -539,17 +473,18 @@ python-versions = ">=3.6" [package.dependencies] certifi = "*" charset-normalizer = "*" -httpcore = ">=0.13.3,<0.14.0" +httpcore = ">=0.14.0,<0.15.0" rfc3986 = {version = ">=1.3,<2", extras = ["idna2008"]} sniffio = "*" [package.extras] brotli = ["brotlicffi", "brotli"] +cli = ["click (>=8.0.0,<9.0.0)", "rich (>=10.0.0,<11.0.0)", "pygments (>=2.0.0,<3.0.0)"] http2 = ["h2 (>=3,<5)"] [[package]] name = "idna" -version = "3.2" +version = "3.3" description = "Internationalized Domain Names in Applications (IDNA)" category = "main" optional = false @@ -557,20 +492,19 @@ python-versions = ">=3.5" [[package]] name = "importlib-metadata" -version = "4.8.1" +version = "4.11.3" description = "Read metadata from Python packages" category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [package.dependencies] -typing-extensions = {version = ">=3.6.4", markers = "python_version < \"3.8\""} zipp = ">=0.5" [package.extras] -docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] +docs = ["sphinx", "jaraco.packaging (>=9)", "rst.linker (>=1.9)"] perf = ["ipython"] -testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pep517", "pyfakefs", "flufl.flake8", "pytest-perf (>=0.9.2)", "pytest-black (>=0.3.7)", "pytest-mypy", "importlib-resources (>=1.3)"] +testing = ["pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "packaging", "pyfakefs", "flufl.flake8", "pytest-perf (>=0.9.2)", "pytest-black (>=0.3.7)", "pytest-mypy (>=0.9.1)", "importlib-resources (>=1.3)"] [[package]] name = "iniconfig" @@ -582,7 +516,7 @@ python-versions = "*" [[package]] name = "isort" -version = "5.9.3" +version = "5.10.1" description = "A Python utility / library to sort Python imports." category = "dev" optional = false @@ -596,7 +530,7 @@ plugins = ["setuptools"] [[package]] name = "jinja2" -version = "3.0.1" +version = "3.0.3" description = "A very fast and expressive template engine." category = "dev" optional = false @@ -608,21 +542,9 @@ MarkupSafe = ">=2.0" [package.extras] i18n = ["Babel (>=2.7)"] -[[package]] -name = "livereload" -version = "2.6.3" -description = "Python LiveReload is an awesome tool for web developers" -category = "dev" -optional = false -python-versions = "*" - -[package.dependencies] -six = "*" -tornado = {version = "*", markers = "python_version > \"2.7\""} - [[package]] name = "loguru" -version = "0.5.3" +version = "0.6.0" description = "Python logging made (stupidly) simple" category = "main" optional = false @@ -633,40 +555,29 @@ colorama = {version = ">=0.3.4", markers = "sys_platform == \"win32\""} win32-setctime = {version = ">=1.0.0", markers = "sys_platform == \"win32\""} [package.extras] -dev = ["codecov (>=2.0.15)", "colorama (>=0.3.4)", "flake8 (>=3.7.7)", "tox (>=3.9.0)", "tox-travis (>=0.12)", "pytest (>=4.6.2)", "pytest-cov (>=2.7.1)", "Sphinx (>=2.2.1)", "sphinx-autobuild (>=0.7.1)", "sphinx-rtd-theme (>=0.4.3)", "black (>=19.10b0)", "isort (>=5.1.1)"] +dev = ["colorama (>=0.3.4)", "docutils (==0.16)", "flake8 (>=3.7.7)", "tox (>=3.9.0)", "pytest (>=4.6.2)", "pytest-cov (>=2.7.1)", "black (>=19.10b0)", "isort (>=5.1.1)", "Sphinx (>=4.1.1)", "sphinx-autobuild (>=0.7.1)", "sphinx-rtd-theme (>=0.4.3)"] [[package]] name = "markdown" -version = "3.3.4" +version = "3.3.6" description = "Python implementation of Markdown." category = "dev" optional = false python-versions = ">=3.6" [package.dependencies] -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} +importlib-metadata = {version = ">=4.4", markers = "python_version < \"3.10\""} [package.extras] testing = ["coverage", "pyyaml"] -[[package]] -name = "markdown-include" -version = "0.6.0" -description = "This is an extension to Python-Markdown which provides an \"include\" function, similar to that found in LaTeX (and also the C pre-processor and Fortran). I originally wrote it for my FORD Fortran auto-documentation generator." -category = "dev" -optional = false -python-versions = "*" - -[package.dependencies] -markdown = "*" - [[package]] name = "markupsafe" -version = "2.0.1" +version = "2.1.1" description = "Safely add untrusted strings to HTML/XML markup." category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [[package]] name = "mccabe" @@ -686,7 +597,7 @@ python-versions = ">=3.6" [[package]] name = "mkdocs" -version = "1.2.2" +version = "1.2.3" description = "Project documentation with Markdown." category = "dev" optional = false @@ -707,32 +618,21 @@ watchdog = ">=2.0" [package.extras] i18n = ["babel (>=2.9.0)"] -[[package]] -name = "mkdocs-autorefs" -version = "0.2.1" -description = "Automatically link across pages in MkDocs." -category = "dev" -optional = false -python-versions = ">=3.6,<4.0" - -[package.dependencies] -Markdown = ">=3.3,<4.0" -mkdocs = ">=1.1,<2.0" - [[package]] name = "mkdocs-material" -version = "7.2.6" +version = "8.1.9" description = "A Material Design theme for MkDocs" category = "dev" optional = false -python-versions = "*" +python-versions = ">=3.6" [package.dependencies] +jinja2 = ">=2.11.1" markdown = ">=3.2" -mkdocs = ">=1.2.2" +mkdocs = ">=1.2.3" mkdocs-material-extensions = ">=1.0" -Pygments = ">=2.4" -pymdown-extensions = ">=7.0" +pygments = ">=2.10" +pymdown-extensions = ">=9.0" [[package]] name = "mkdocs-material-extensions" @@ -742,42 +642,9 @@ category = "dev" optional = false python-versions = ">=3.6" -[[package]] -name = "mkdocstrings" -version = "0.15.2" -description = "Automatic documentation from sources, for MkDocs." -category = "dev" -optional = false -python-versions = ">=3.6,<4.0" - -[package.dependencies] -Jinja2 = ">=2.11.1,<4.0" -Markdown = ">=3.3,<4.0" -MarkupSafe = ">=1.1,<3.0" -mkdocs = ">=1.1.1,<2.0.0" -mkdocs-autorefs = ">=0.1,<0.3" -pymdown-extensions = ">=6.3,<9.0" -pytkdocs = ">=0.2.0,<0.12.0" - -[[package]] -name = "molten" -version = "1.0.2" -description = "A minimal, extensible, fast and productive API framework." -category = "main" -optional = true -python-versions = ">=3.6" - -[package.dependencies] -typing-extensions = ">=3.6,<4.0" -typing-inspect = ">=0.3.1,<0.7" - -[package.extras] -all = ["typing-extensions (>=3.6,<4.0)", "typing-inspect (>=0.3.1,<0.7)"] -dev = ["typing-extensions (>=3.6,<4.0)", "typing-inspect (>=0.3.1,<0.7)", "alabaster (>0.7)", "sphinx (<1.8)", "sphinxcontrib-napoleon", "flake8", "flake8-bugbear", "flake8-quotes", "isort", "mypy", "bumpversion (>0.5,<0.6)", "dramatiq[rabbitmq] (>1.3,<2.0)", "gevent", "gunicorn (>19.8)", "jinja2 (>=2.10,<3.0)", "msgpack (>0.5,<0.6)", "prometheus-client (>=0.2,<0.3)", "sqlalchemy (>1.2,<2.0)", "toml (>0.9,<0.10)", "wsgicors (>=0.7,<0.8)", "pytest"] - [[package]] name = "mypy" -version = "0.812" +version = "0.910" description = "Optional static typing for Python" category = "dev" optional = false @@ -785,11 +652,12 @@ python-versions = ">=3.5" [package.dependencies] mypy-extensions = ">=0.4.3,<0.5.0" -typed-ast = ">=1.4.0,<1.5.0" +toml = "*" typing-extensions = ">=3.7.4" [package.extras] dmypy = ["psutil (>=4.0)"] +python2 = ["typed-ast (>=1.4.0,<1.5.0)"] [[package]] name = "mypy-extensions" @@ -799,35 +667,16 @@ category = "main" optional = false python-versions = "*" -[[package]] -name = "nox" -version = "2021.6.12" -description = "Flexible test automation." -category = "dev" -optional = false -python-versions = ">=3.6" - -[package.dependencies] -argcomplete = ">=1.9.4,<2.0" -colorlog = ">=2.6.1,<7.0.0" -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} -packaging = ">=20.9" -py = ">=1.4.0,<2.0.0" -virtualenv = ">=14.0.0" - -[package.extras] -tox_to_nox = ["jinja2", "tox"] - [[package]] name = "packaging" -version = "21.0" +version = "21.3" description = "Core utilities for Python packages" category = "dev" optional = false python-versions = ">=3.6" [package.dependencies] -pyparsing = ">=2.0.2" +pyparsing = ">=2.0.2,<3.0.5 || >3.0.5" [[package]] name = "pathspec" @@ -839,7 +688,7 @@ python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" [[package]] name = "pbr" -version = "5.6.0" +version = "5.8.1" description = "Python Build Reasonableness" category = "dev" optional = false @@ -847,22 +696,23 @@ python-versions = ">=2.6" [[package]] name = "pep8-naming" -version = "0.11.1" +version = "0.12.1" description = "Check PEP-8 naming conventions, plugin for flake8" category = "dev" optional = false python-versions = "*" [package.dependencies] +flake8 = ">=3.9.1" flake8-polyfill = ">=1.0.2,<2" [[package]] name = "platformdirs" -version = "2.3.0" +version = "2.5.1" description = "A small Python module for determining appropriate platform-specific dirs, e.g. a \"user data dir\"." category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [package.extras] docs = ["Sphinx (>=4)", "furo (>=2021.7.5b38)", "proselint (>=0.10.2)", "sphinx-autodoc-typehints (>=1.12)"] @@ -870,33 +720,31 @@ test = ["appdirs (==1.4.4)", "pytest (>=6)", "pytest-cov (>=2.7)", "pytest-mock [[package]] name = "pluggy" -version = "0.13.1" +version = "1.0.0" description = "plugin and hook calling mechanisms for python" category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" - -[package.dependencies] -importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""} +python-versions = ">=3.6" [package.extras] dev = ["pre-commit", "tox"] +testing = ["pytest", "pytest-benchmark"] [[package]] name = "py" -version = "1.10.0" +version = "1.11.0" description = "library with cross-python path, ini-parsing, io, code, log facilities" category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" [[package]] name = "pycodestyle" -version = "2.7.0" +version = "2.8.0" description = "Python style guide checker" category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" [[package]] name = "pydantic" @@ -929,7 +777,7 @@ toml = ["toml"] [[package]] name = "pyflakes" -version = "2.3.1" +version = "2.4.0" description = "passive checker of Python programs" category = "dev" optional = false @@ -937,7 +785,7 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" [[package]] name = "pygments" -version = "2.10.0" +version = "2.11.2" description = "Pygments is a syntax highlighting package written in Python." category = "dev" optional = false @@ -945,22 +793,25 @@ python-versions = ">=3.5" [[package]] name = "pymdown-extensions" -version = "8.2" +version = "9.3" description = "Extension pack for Python Markdown." category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [package.dependencies] Markdown = ">=3.2" [[package]] name = "pyparsing" -version = "2.4.7" +version = "3.0.7" description = "Python parsing module" category = "dev" optional = false -python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" +python-versions = ">=3.6" + +[package.extras] +diagrams = ["jinja2", "railroad-diagrams"] [[package]] name = "pytest" @@ -974,7 +825,6 @@ python-versions = ">=3.6" atomicwrites = {version = ">=1.0", markers = "sys_platform == \"win32\""} attrs = ">=19.2.0" colorama = {version = "*", markers = "sys_platform == \"win32\""} -importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""} iniconfig = "*" packaging = "*" pluggy = ">=0.12,<2.0" @@ -986,7 +836,7 @@ testing = ["argcomplete", "hypothesis (>=3.56)", "mock", "nose", "requests", "xm [[package]] name = "pytest-asyncio" -version = "0.15.1" +version = "0.16.0" description = "Pytest support for asyncio." category = "dev" optional = false @@ -998,30 +848,17 @@ pytest = ">=5.4.0" [package.extras] testing = ["coverage", "hypothesis (>=5.7.1)"] -[[package]] -name = "pytest-clarity" -version = "0.3.0a0" -description = "A plugin providing an alternative, colourful diff output for failing assertions." -category = "dev" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" - -[package.dependencies] -pytest = ">=3.5.0" -termcolor = "1.1.0" - [[package]] name = "pytest-cov" -version = "2.12.1" +version = "3.0.0" description = "Pytest plugin for measuring coverage." category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" +python-versions = ">=3.6" [package.dependencies] -coverage = ">=5.2.1" +coverage = {version = ">=5.2.1", extras = ["toml"]} pytest = ">=4.6" -toml = "*" [package.extras] testing = ["fields", "hunter", "process-tests", "six", "pytest-xdist", "virtualenv"] @@ -1038,31 +875,23 @@ python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7" six = ">=1.5" [[package]] -name = "python-multipart" -version = "0.0.5" -description = "A streaming multipart parser for Python" -category = "main" -optional = true -python-versions = "*" - -[package.dependencies] -six = ">=1.4.0" - -[[package]] -name = "pytkdocs" -version = "0.7.0" -description = "Load Python objects documentation." +name = "python-dotenv" +version = "0.19.2" +description = "Read key-value pairs from a .env file and set them as environment variables" category = "dev" optional = false -python-versions = ">=3.6,<4.0" +python-versions = ">=3.5" + +[package.extras] +cli = ["click (>=5.0)"] [[package]] name = "pyyaml" -version = "5.4.1" +version = "6.0" description = "YAML parser and emitter for Python" category = "dev" optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" +python-versions = ">=3.6" [[package]] name = "pyyaml-env-tag" @@ -1076,16 +905,37 @@ python-versions = ">=3.6" pyyaml = "*" [[package]] -name = "regex" -version = "2021.8.28" -description = "Alternative regular expression module, to replace re." +name = "requests" +version = "2.26.0" +description = "Python HTTP for Humans." category = "dev" optional = false -python-versions = "*" +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, !=3.5.*" + +[package.dependencies] +certifi = ">=2017.4.17" +charset-normalizer = {version = ">=2.0.0,<2.1.0", markers = "python_version >= \"3\""} +idna = {version = ">=2.5,<4", markers = "python_version >= \"3\""} +urllib3 = ">=1.21.1,<1.27" + +[package.extras] +socks = ["PySocks (>=1.5.6,!=1.5.7)", "win-inet-pton"] +use_chardet_on_py3 = ["chardet (>=3.0.2,<5)"] + +[[package]] +name = "respx" +version = "0.19.0" +description = "A utility for mocking out the Python HTTPX and HTTP Core libraries." +category = "dev" +optional = false +python-versions = ">=3.6" + +[package.dependencies] +httpx = ">=0.21.0" [[package]] name = "restructuredtext-lint" -version = "1.3.2" +version = "1.4.0" description = "reStructuredText linter" category = "dev" optional = false @@ -1112,17 +962,17 @@ idna2008 = ["idna"] name = "six" version = "1.16.0" description = "Python 2 and 3 compatibility utilities" -category = "main" +category = "dev" optional = false python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*" [[package]] name = "smmap" -version = "4.0.0" +version = "5.0.0" description = "A pure Python implementation of a sliding window memory map manager" category = "dev" optional = false -python-versions = ">=3.5" +python-versions = ">=3.6" [[package]] name = "sniffio" @@ -1134,7 +984,7 @@ python-versions = ">=3.5" [[package]] name = "snowballstemmer" -version = "2.1.0" +version = "2.2.0" description = "This package provides 29 stemmers for 28 languages generated from Snowball algorithms." category = "dev" optional = false @@ -1142,7 +992,7 @@ python-versions = "*" [[package]] name = "starlette" -version = "0.16.0" +version = "0.17.1" description = "The little ASGI library that shines." category = "dev" optional = false @@ -1150,34 +1000,24 @@ python-versions = ">=3.6" [package.dependencies] anyio = ">=3.0.0,<4" -typing-extensions = {version = "*", markers = "python_version < \"3.8\""} [package.extras] -full = ["itsdangerous", "jinja2", "python-multipart", "pyyaml", "requests", "graphene"] +full = ["itsdangerous", "jinja2", "python-multipart", "pyyaml", "requests"] [[package]] name = "stevedore" -version = "3.4.0" +version = "3.5.0" description = "Manage dynamic plugins for Python applications" category = "dev" optional = false python-versions = ">=3.6" [package.dependencies] -importlib-metadata = {version = ">=1.7.0", markers = "python_version < \"3.8\""} pbr = ">=2.0.0,<2.1.0 || >2.1.0" -[[package]] -name = "termcolor" -version = "1.1.0" -description = "ANSII Color formatting for output in terminal." -category = "dev" -optional = false -python-versions = "*" - [[package]] name = "testfixtures" -version = "6.18.1" +version = "6.18.5" description = "A collection of helpers and mock objects for unit tests and doc tests." category = "dev" optional = false @@ -1190,7 +1030,7 @@ test = ["pytest (>=3.6)", "pytest-cov", "pytest-django", "zope.component", "sybi [[package]] name = "tokenize-rt" -version = "4.1.0" +version = "4.2.1" description = "A wrapper around the stdlib `tokenize` which roundtrips." category = "dev" optional = false @@ -1205,75 +1045,64 @@ optional = false python-versions = ">=2.6, !=3.0.*, !=3.1.*, !=3.2.*" [[package]] -name = "tornado" -version = "6.1" -description = "Tornado is a Python web framework and asynchronous networking library, originally developed at FriendFeed." +name = "tomli" +version = "1.2.3" +description = "A lil' TOML parser" category = "dev" optional = false -python-versions = ">= 3.5" - -[[package]] -name = "typed-ast" -version = "1.4.3" -description = "a fork of Python 2 and 3 ast modules with type comment support" -category = "dev" -optional = false -python-versions = "*" +python-versions = ">=3.6" [[package]] name = "typing-extensions" -version = "3.10.0.2" -description = "Backported and Experimental Type Hints for Python 3.5+" +version = "4.1.1" +description = "Backported and Experimental Type Hints for Python 3.6+" category = "main" optional = false -python-versions = "*" +python-versions = ">=3.6" [[package]] -name = "typing-inspect" -version = "0.6.0" -description = "Runtime inspection utilities for typing module." -category = "main" -optional = true -python-versions = "*" +name = "urllib3" +version = "1.26.9" +description = "HTTP library with thread-safe connection pooling, file post, and more." +category = "dev" +optional = false +python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*, <4" -[package.dependencies] -mypy-extensions = ">=0.3.0" -typing-extensions = ">=3.7.4" +[package.extras] +brotli = ["brotlicffi (>=0.8.0)", "brotli (>=1.0.9)", "brotlipy (>=0.6.0)"] +secure = ["pyOpenSSL (>=0.14)", "cryptography (>=1.3.4)", "idna (>=2.0.0)", "certifi", "ipaddress"] +socks = ["PySocks (>=1.5.6,!=1.5.7,<2.0)"] [[package]] -name = "virtualenv" -version = "20.7.2" -description = "Virtual Python Environment builder" +name = "uvicorn" +version = "0.16.0" +description = "The lightning-fast ASGI server." category = "dev" optional = false -python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,>=2.7" +python-versions = "*" [package.dependencies] -"backports.entry-points-selectable" = ">=1.0.4" -distlib = ">=0.3.1,<1" -filelock = ">=3.0.0,<4" -importlib-metadata = {version = ">=0.12", markers = "python_version < \"3.8\""} -platformdirs = ">=2,<3" -six = ">=1.9.0,<2" +asgiref = ">=3.4.0" +click = ">=7.0" +h11 = ">=0.8" [package.extras] -docs = ["proselint (>=0.10.2)", "sphinx (>=3)", "sphinx-argparse (>=0.2.5)", "sphinx-rtd-theme (>=0.4.3)", "towncrier (>=19.9.0rc1)"] -testing = ["coverage (>=4)", "coverage-enable-subprocess (>=1)", "flaky (>=3)", "pytest (>=4)", "pytest-env (>=0.6.2)", "pytest-freezegun (>=0.4.1)", "pytest-mock (>=2)", "pytest-randomly (>=1)", "pytest-timeout (>=1)", "packaging (>=20.0)"] +standard = ["httptools (>=0.2.0,<0.4.0)", "watchgod (>=0.6)", "python-dotenv (>=0.13)", "PyYAML (>=5.1)", "websockets (>=9.1)", "websockets (>=10.0)", "uvloop (>=0.14.0,!=0.15.0,!=0.15.1)", "colorama (>=0.4)"] [[package]] name = "watchdog" -version = "2.1.5" +version = "2.1.6" description = "Filesystem events monitoring" category = "dev" optional = false python-versions = ">=3.6" [package.extras] -watchmedo = ["PyYAML (>=3.10)", "argh (>=0.24.1)"] +watchmedo = ["PyYAML (>=3.10)"] [[package]] name = "wemake-python-styleguide" -version = "0.15.3" +version = "0.16.0" description = "The strictest and most opinionated python linter ever" category = "dev" optional = false @@ -1283,9 +1112,9 @@ python-versions = ">=3.6,<4.0" astor = ">=0.8,<0.9" attrs = "*" darglint = ">=1.2,<2.0" -flake8 = ">=3.7,<4.0" +flake8 = ">=3.7,<5" flake8-bandit = ">=2.1,<3.0" -flake8-broken-line = ">=0.3,<0.4" +flake8-broken-line = ">=0.3,<0.5" flake8-bugbear = ">=20.1,<22.0" flake8-commas = ">=2.0,<3.0" flake8-comprehensions = ">=3.1,<4.0" @@ -1296,14 +1125,13 @@ flake8-isort = ">=4.0,<5.0" flake8-quotes = ">=3.0,<4.0" flake8-rst-docstrings = ">=0.2.3,<0.3.0" flake8-string-format = ">=0.3,<0.4" -importlib-metadata = {version = "*", markers = "python_version < \"3.8\""} -pep8-naming = ">=0.11,<0.12" +pep8-naming = ">=0.11,<0.13" pygments = ">=2.4,<3.0" -typing_extensions = ">=3.6,<4.0" +typing_extensions = ">=3.6,<5.0" [[package]] name = "win32-setctime" -version = "1.0.3" +version = "1.1.0" description = "A small Python utility to set file creation time on Windows" category = "main" optional = false @@ -1314,44 +1142,37 @@ dev = ["pytest (>=4.6.2)", "black (>=19.3b0)"] [[package]] name = "zipp" -version = "3.5.0" +version = "3.7.0" description = "Backport of pathlib-compatible object wrapper for zip files" category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" [package.extras] docs = ["sphinx", "jaraco.packaging (>=8.2)", "rst.linker (>=1.9)"] -testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"] - -[extras] -tests = ["aiofiles", "molten", "python-multipart"] +testing = ["pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytest-cov", "pytest-enabler (>=1.0.1)", "jaraco.itertools", "func-timeout", "pytest-black (>=0.3.7)", "pytest-mypy"] [metadata] lock-version = "1.1" -python-versions = "^3.7" -content-hash = "4f4ed47faa04787bd89453652c1cf77f4e419feafba93388c15cab12c1b5d6b1" +python-versions = ">=3.8,<3.11" +content-hash = "29869ce33ae5ceefbd9ee52a0ad98196ea521d93607f83421d5494f88c63e2aa" [metadata.files] add-trailing-comma = [ - {file = "add_trailing_comma-2.1.0-py2.py3-none-any.whl", hash = "sha256:f462403aa2e997e20855708edb57536d1d3310d5c5fac7e80542578eb47fdb10"}, - {file = "add_trailing_comma-2.1.0.tar.gz", hash = "sha256:f9864ffbc12ea4e54916a356d57341ab58f612867c2ad453339c51004807e8ce"}, + {file = "add_trailing_comma-2.2.1-py2.py3-none-any.whl", hash = "sha256:981c18282b38ec5bceab80ef11485440334d2a274fcf3fce1f91692374b6d818"}, + {file = "add_trailing_comma-2.2.1.tar.gz", hash = "sha256:1640e97c4e85132633a6cb19b29e392dbaf9516292388afa685f7ef1012468e0"}, ] aiofiles = [ - {file = "aiofiles-0.7.0-py3-none-any.whl", hash = "sha256:c67a6823b5f23fcab0a2595a289cec7d8c863ffcb4322fb8cd6b90400aedfdbc"}, - {file = "aiofiles-0.7.0.tar.gz", hash = "sha256:a1c4fc9b2ff81568c83e21392a82f344ea9d23da906e4f6a52662764545e19d4"}, + {file = "aiofiles-0.8.0-py3-none-any.whl", hash = "sha256:7a973fc22b29e9962d0897805ace5856e6a566ab1f0c8e5c91ff6c866519c937"}, + {file = "aiofiles-0.8.0.tar.gz", hash = "sha256:8334f23235248a3b2e83b2c3a78a22674f39969b96397126cc93664d9a901e59"}, ] anyio = [ - {file = "anyio-3.3.1-py3-none-any.whl", hash = "sha256:d7c604dd491eca70e19c78664d685d5e4337612d574419d503e76f5d7d1590bd"}, - {file = "anyio-3.3.1.tar.gz", hash = "sha256:85913b4e2fec030e8c72a8f9f98092eeb9e25847a6e00d567751b77e34f856fe"}, -] -appdirs = [ - {file = "appdirs-1.4.4-py2.py3-none-any.whl", hash = "sha256:a841dacd6b99318a741b166adb07e19ee71a274450e68237b4650ca1055ab128"}, - {file = "appdirs-1.4.4.tar.gz", hash = "sha256:7d5d0167b2b1ba821647616af46a749d1c653740dd0d2415100fe26e27afdf41"}, + {file = "anyio-3.5.0-py3-none-any.whl", hash = "sha256:b5fa16c5ff93fa1046f2eeb5bbff2dad4d3514d6cda61d02816dba34fa8c3c2e"}, + {file = "anyio-3.5.0.tar.gz", hash = "sha256:a0aeffe2fb1fdf374a8e4b471444f0f3ac4fb9f5a5b542b48824475e0042a5a6"}, ] -argcomplete = [ - {file = "argcomplete-1.12.3-py2.py3-none-any.whl", hash = "sha256:291f0beca7fd49ce285d2f10e4c1c77e9460cf823eef2de54df0c0fec88b0d81"}, - {file = "argcomplete-1.12.3.tar.gz", hash = "sha256:2c7dbffd8c045ea534921e63b0be6fe65e88599990d8dc408ac8c542b72a5445"}, +asgiref = [ + {file = "asgiref-3.5.0-py3-none-any.whl", hash = "sha256:88d59c13d634dcffe0510be048210188edd79aeccb6a6c9028cdad6f31d730a9"}, + {file = "asgiref-3.5.0.tar.gz", hash = "sha256:2f8abc20f7248433085eda803936d98992f1343ddb022065779f37c5da0181d0"}, ] astor = [ {file = "astor-0.8.1-py2.py3-none-any.whl", hash = "sha256:070a54e890cefb5b3739d19f30f5a5ec840ffc9c50ffa7d23cc9fc1a38ebbfc5"}, @@ -1362,146 +1183,116 @@ atomicwrites = [ {file = "atomicwrites-1.4.0.tar.gz", hash = "sha256:ae70396ad1a434f9c7046fd2dd196fc04b12f9e91ffb859164193be8b6168a7a"}, ] attrs = [ - {file = "attrs-21.2.0-py2.py3-none-any.whl", hash = "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1"}, - {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"}, + {file = "attrs-21.4.0-py2.py3-none-any.whl", hash = "sha256:2d27e3784d7a565d36ab851fe94887c5eccd6a463168875832a1be79c82828b4"}, + {file = "attrs-21.4.0.tar.gz", hash = "sha256:626ba8234211db98e869df76230a137c4c40a12d72445c45d5f5b716f076e2fd"}, ] autoflake = [ {file = "autoflake-1.4.tar.gz", hash = "sha256:61a353012cff6ab94ca062823d1fb2f692c4acda51c76ff83a8d77915fba51ea"}, ] -"backports.entry-points-selectable" = [ - {file = "backports.entry_points_selectable-1.1.0-py2.py3-none-any.whl", hash = "sha256:a6d9a871cde5e15b4c4a53e3d43ba890cc6861ec1332c9c2428c92f977192acc"}, - {file = "backports.entry_points_selectable-1.1.0.tar.gz", hash = "sha256:988468260ec1c196dab6ae1149260e2f5472c9110334e5d51adcb77867361f6a"}, -] bandit = [ - {file = "bandit-1.7.0-py3-none-any.whl", hash = "sha256:216be4d044209fa06cf2a3e51b319769a51be8318140659719aa7a115c35ed07"}, - {file = "bandit-1.7.0.tar.gz", hash = "sha256:8a4c7415254d75df8ff3c3b15cfe9042ecee628a1e40b44c15a98890fbfc2608"}, -] -base64io = [ - {file = "base64io-1.0.3-py2.py3-none-any.whl", hash = "sha256:e9a6c9f470e34f8debaad26134bcf3f0bcbf677dac73e32295cfb2915d30815b"}, - {file = "base64io-1.0.3.tar.gz", hash = "sha256:24f2d0fe765c35339e1b2d33aa95f9137b1b765b594164fad1016c15827a7073"}, + {file = "bandit-1.7.2-py3-none-any.whl", hash = "sha256:e20402cadfd126d85b68ed4c8862959663c8c372dbbb1fca8f8e2c9f55a067ec"}, + {file = "bandit-1.7.2.tar.gz", hash = "sha256:6d11adea0214a43813887bfe71a377b5a9955e4c826c8ffd341b494e3ab25260"}, ] black = [ - {file = "black-20.8b1.tar.gz", hash = "sha256:1c02557aa099101b9d21496f8a914e9ed2222ef70336404eeeac8edba836fbea"}, + {file = "black-21.12b0-py3-none-any.whl", hash = "sha256:a615e69ae185e08fdd73e4715e260e2479c861b5740057fde6e8b4e3b7dd589f"}, + {file = "black-21.12b0.tar.gz", hash = "sha256:77b80f693a569e2e527958459634f18df9b0ba2625ba4e0c2d5da5be42e6f2b3"}, ] certifi = [ - {file = "certifi-2021.5.30-py2.py3-none-any.whl", hash = "sha256:50b1e4f8446b06f41be7dd6338db18e0990601dce795c2b1686458aa7e8fa7d8"}, - {file = "certifi-2021.5.30.tar.gz", hash = "sha256:2bbf76fd432960138b3ef6dda3dde0544f27cbf8546c458e60baf371917ba9ee"}, + {file = "certifi-2021.10.8-py2.py3-none-any.whl", hash = "sha256:d62a0163eb4c2344ac042ab2bdf75399a71a2d8c7d47eac2e2ee91b9d6339569"}, + {file = "certifi-2021.10.8.tar.gz", hash = "sha256:78884e7c1d4b00ce3cea67b44566851c4343c120abd683433ce934a68ea58872"}, ] charset-normalizer = [ - {file = "charset-normalizer-2.0.4.tar.gz", hash = "sha256:f23667ebe1084be45f6ae0538e4a5a865206544097e4e8bbcacf42cd02a348f3"}, - {file = "charset_normalizer-2.0.4-py3-none-any.whl", hash = "sha256:0c8911edd15d19223366a194a513099a302055a962bca2cec0f54b8b63175d8b"}, + {file = "charset-normalizer-2.0.12.tar.gz", hash = "sha256:2857e29ff0d34db842cd7ca3230549d1a697f96ee6d3fb071cfa6c7393832597"}, + {file = "charset_normalizer-2.0.12-py3-none-any.whl", hash = "sha256:6881edbebdb17b39b4eaaa821b438bf6eddffb4468cf344f09f89def34a8b1df"}, ] click = [ - {file = "click-8.0.1-py3-none-any.whl", hash = "sha256:fba402a4a47334742d782209a7c79bc448911afe1149d07bdabdf480b3e2f4b6"}, - {file = "click-8.0.1.tar.gz", hash = "sha256:8c04c11192119b1ef78ea049e0a6f0463e4c48ef00a30160c704337586f3ad7a"}, + {file = "click-8.0.4-py3-none-any.whl", hash = "sha256:6a7a62563bbfabfda3a38f3023a1db4a35978c0abd76f6c9605ecd6554d6d9b1"}, + {file = "click-8.0.4.tar.gz", hash = "sha256:8458d7b1287c5fb128c90e23381cf99dcde74beaf6c7ff6384ce84d6fe090adb"}, ] colorama = [ {file = "colorama-0.4.4-py2.py3-none-any.whl", hash = "sha256:9f47eda37229f68eee03b24b9748937c7dc3868f906e8ba69fbcbdd3bc5dc3e2"}, {file = "colorama-0.4.4.tar.gz", hash = "sha256:5941b2b48a20143d2267e95b1c2a7603ce057ee39fd88e7329b0c292aa16869b"}, ] -colorlog = [ - {file = "colorlog-4.8.0-py2.py3-none-any.whl", hash = "sha256:3dd15cb27e8119a24c1a7b5c93f9f3b455855e0f73993b1c25921b2f646f1dcd"}, - {file = "colorlog-4.8.0.tar.gz", hash = "sha256:59b53160c60902c405cdec28d38356e09d40686659048893e026ecbd589516b1"}, -] coverage = [ - {file = "coverage-5.5-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:b6d534e4b2ab35c9f93f46229363e17f63c53ad01330df9f2d6bd1187e5eaacf"}, - {file = "coverage-5.5-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:b7895207b4c843c76a25ab8c1e866261bcfe27bfaa20c192de5190121770672b"}, - {file = "coverage-5.5-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:c2723d347ab06e7ddad1a58b2a821218239249a9e4365eaff6649d31180c1669"}, - {file = "coverage-5.5-cp27-cp27m-manylinux2010_i686.whl", hash = "sha256:900fbf7759501bc7807fd6638c947d7a831fc9fdf742dc10f02956ff7220fa90"}, - {file = "coverage-5.5-cp27-cp27m-manylinux2010_x86_64.whl", hash = "sha256:004d1880bed2d97151facef49f08e255a20ceb6f9432df75f4eef018fdd5a78c"}, - {file = "coverage-5.5-cp27-cp27m-win32.whl", hash = "sha256:06191eb60f8d8a5bc046f3799f8a07a2d7aefb9504b0209aff0b47298333302a"}, - {file = "coverage-5.5-cp27-cp27m-win_amd64.whl", hash = "sha256:7501140f755b725495941b43347ba8a2777407fc7f250d4f5a7d2a1050ba8e82"}, - {file = "coverage-5.5-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:372da284cfd642d8e08ef606917846fa2ee350f64994bebfbd3afb0040436905"}, - {file = "coverage-5.5-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:8963a499849a1fc54b35b1c9f162f4108017b2e6db2c46c1bed93a72262ed083"}, - {file = "coverage-5.5-cp27-cp27mu-manylinux2010_i686.whl", hash = "sha256:869a64f53488f40fa5b5b9dcb9e9b2962a66a87dab37790f3fcfb5144b996ef5"}, - {file = "coverage-5.5-cp27-cp27mu-manylinux2010_x86_64.whl", hash = "sha256:4a7697d8cb0f27399b0e393c0b90f0f1e40c82023ea4d45d22bce7032a5d7b81"}, - {file = "coverage-5.5-cp310-cp310-macosx_10_14_x86_64.whl", hash = "sha256:8d0a0725ad7c1a0bcd8d1b437e191107d457e2ec1084b9f190630a4fb1af78e6"}, - {file = "coverage-5.5-cp310-cp310-manylinux1_x86_64.whl", hash = "sha256:51cb9476a3987c8967ebab3f0fe144819781fca264f57f89760037a2ea191cb0"}, - {file = "coverage-5.5-cp310-cp310-win_amd64.whl", hash = "sha256:c0891a6a97b09c1f3e073a890514d5012eb256845c451bd48f7968ef939bf4ae"}, - {file = "coverage-5.5-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:3487286bc29a5aa4b93a072e9592f22254291ce96a9fbc5251f566b6b7343cdb"}, - {file = "coverage-5.5-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:deee1077aae10d8fa88cb02c845cfba9b62c55e1183f52f6ae6a2df6a2187160"}, - {file = "coverage-5.5-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:f11642dddbb0253cc8853254301b51390ba0081750a8ac03f20ea8103f0c56b6"}, - {file = "coverage-5.5-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:6c90e11318f0d3c436a42409f2749ee1a115cd8b067d7f14c148f1ce5574d701"}, - {file = "coverage-5.5-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:30c77c1dc9f253283e34c27935fded5015f7d1abe83bc7821680ac444eaf7793"}, - {file = "coverage-5.5-cp35-cp35m-win32.whl", hash = "sha256:9a1ef3b66e38ef8618ce5fdc7bea3d9f45f3624e2a66295eea5e57966c85909e"}, - {file = "coverage-5.5-cp35-cp35m-win_amd64.whl", hash = "sha256:972c85d205b51e30e59525694670de6a8a89691186012535f9d7dbaa230e42c3"}, - {file = "coverage-5.5-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:af0e781009aaf59e25c5a678122391cb0f345ac0ec272c7961dc5455e1c40066"}, - {file = "coverage-5.5-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:74d881fc777ebb11c63736622b60cb9e4aee5cace591ce274fb69e582a12a61a"}, - {file = "coverage-5.5-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:92b017ce34b68a7d67bd6d117e6d443a9bf63a2ecf8567bb3d8c6c7bc5014465"}, - {file = "coverage-5.5-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:d636598c8305e1f90b439dbf4f66437de4a5e3c31fdf47ad29542478c8508bbb"}, - {file = "coverage-5.5-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:41179b8a845742d1eb60449bdb2992196e211341818565abded11cfa90efb821"}, - {file = "coverage-5.5-cp36-cp36m-win32.whl", hash = "sha256:040af6c32813fa3eae5305d53f18875bedd079960822ef8ec067a66dd8afcd45"}, - {file = "coverage-5.5-cp36-cp36m-win_amd64.whl", hash = "sha256:5fec2d43a2cc6965edc0bb9e83e1e4b557f76f843a77a2496cbe719583ce8184"}, - {file = "coverage-5.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:18ba8bbede96a2c3dde7b868de9dcbd55670690af0988713f0603f037848418a"}, - {file = "coverage-5.5-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:2910f4d36a6a9b4214bb7038d537f015346f413a975d57ca6b43bf23d6563b53"}, - {file = "coverage-5.5-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:f0b278ce10936db1a37e6954e15a3730bea96a0997c26d7fee88e6c396c2086d"}, - {file = "coverage-5.5-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:796c9c3c79747146ebd278dbe1e5c5c05dd6b10cc3bcb8389dfdf844f3ead638"}, - {file = "coverage-5.5-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:53194af30d5bad77fcba80e23a1441c71abfb3e01192034f8246e0d8f99528f3"}, - {file = "coverage-5.5-cp37-cp37m-win32.whl", hash = "sha256:184a47bbe0aa6400ed2d41d8e9ed868b8205046518c52464fde713ea06e3a74a"}, - {file = "coverage-5.5-cp37-cp37m-win_amd64.whl", hash = "sha256:2949cad1c5208b8298d5686d5a85b66aae46d73eec2c3e08c817dd3513e5848a"}, - {file = "coverage-5.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:217658ec7187497e3f3ebd901afdca1af062b42cfe3e0dafea4cced3983739f6"}, - {file = "coverage-5.5-cp38-cp38-manylinux1_i686.whl", hash = "sha256:1aa846f56c3d49205c952d8318e76ccc2ae23303351d9270ab220004c580cfe2"}, - {file = "coverage-5.5-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:24d4a7de75446be83244eabbff746d66b9240ae020ced65d060815fac3423759"}, - {file = "coverage-5.5-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:d1f8bf7b90ba55699b3a5e44930e93ff0189aa27186e96071fac7dd0d06a1873"}, - {file = "coverage-5.5-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:970284a88b99673ccb2e4e334cfb38a10aab7cd44f7457564d11898a74b62d0a"}, - {file = "coverage-5.5-cp38-cp38-win32.whl", hash = "sha256:01d84219b5cdbfc8122223b39a954820929497a1cb1422824bb86b07b74594b6"}, - {file = "coverage-5.5-cp38-cp38-win_amd64.whl", hash = "sha256:2e0d881ad471768bf6e6c2bf905d183543f10098e3b3640fc029509530091502"}, - {file = "coverage-5.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d1f9ce122f83b2305592c11d64f181b87153fc2c2bbd3bb4a3dde8303cfb1a6b"}, - {file = "coverage-5.5-cp39-cp39-manylinux1_i686.whl", hash = "sha256:13c4ee887eca0f4c5a247b75398d4114c37882658300e153113dafb1d76de529"}, - {file = "coverage-5.5-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:52596d3d0e8bdf3af43db3e9ba8dcdaac724ba7b5ca3f6358529d56f7a166f8b"}, - {file = "coverage-5.5-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:2cafbbb3af0733db200c9b5f798d18953b1a304d3f86a938367de1567f4b5bff"}, - {file = "coverage-5.5-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:44d654437b8ddd9eee7d1eaee28b7219bec228520ff809af170488fd2fed3e2b"}, - {file = "coverage-5.5-cp39-cp39-win32.whl", hash = "sha256:d314ed732c25d29775e84a960c3c60808b682c08d86602ec2c3008e1202e3bb6"}, - {file = "coverage-5.5-cp39-cp39-win_amd64.whl", hash = "sha256:13034c4409db851670bc9acd836243aeee299949bd5673e11844befcb0149f03"}, - {file = "coverage-5.5-pp36-none-any.whl", hash = "sha256:f030f8873312a16414c0d8e1a1ddff2d3235655a2174e3648b4fa66b3f2f1079"}, - {file = "coverage-5.5-pp37-none-any.whl", hash = "sha256:2a3859cb82dcbda1cfd3e6f71c27081d18aa251d20a17d87d26d4cd216fb0af4"}, - {file = "coverage-5.5.tar.gz", hash = "sha256:ebe78fe9a0e874362175b02371bdfbee64d8edc42a044253ddf4ee7d3c15212c"}, -] -coverage-conditional-plugin = [ - {file = "coverage-conditional-plugin-0.4.0.tar.gz", hash = "sha256:3ca50bb03b32f8ba8cf63b18d830f334cb75f401579040dd5c61df1c4687fef0"}, - {file = "coverage_conditional_plugin-0.4.0-py3-none-any.whl", hash = "sha256:3bf586f7b793a9e4e2950f7af32711012929b39d1a4b53a4829489e763fda2dc"}, + {file = "coverage-6.3.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:9b27d894748475fa858f9597c0ee1d4829f44683f3813633aaf94b19cb5453cf"}, + {file = "coverage-6.3.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:37d1141ad6b2466a7b53a22e08fe76994c2d35a5b6b469590424a9953155afac"}, + {file = "coverage-6.3.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f9987b0354b06d4df0f4d3e0ec1ae76d7ce7cbca9a2f98c25041eb79eec766f1"}, + {file = "coverage-6.3.2-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:26e2deacd414fc2f97dd9f7676ee3eaecd299ca751412d89f40bc01557a6b1b4"}, + {file = "coverage-6.3.2-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4dd8bafa458b5c7d061540f1ee9f18025a68e2d8471b3e858a9dad47c8d41903"}, + {file = "coverage-6.3.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:46191097ebc381fbf89bdce207a6c107ac4ec0890d8d20f3360345ff5976155c"}, + {file = "coverage-6.3.2-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:6f89d05e028d274ce4fa1a86887b071ae1755082ef94a6740238cd7a8178804f"}, + {file = "coverage-6.3.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:58303469e9a272b4abdb9e302a780072c0633cdcc0165db7eec0f9e32f901e05"}, + {file = "coverage-6.3.2-cp310-cp310-win32.whl", hash = "sha256:2fea046bfb455510e05be95e879f0e768d45c10c11509e20e06d8fcaa31d9e39"}, + {file = "coverage-6.3.2-cp310-cp310-win_amd64.whl", hash = "sha256:a2a8b8bcc399edb4347a5ca8b9b87e7524c0967b335fbb08a83c8421489ddee1"}, + {file = "coverage-6.3.2-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:f1555ea6d6da108e1999b2463ea1003fe03f29213e459145e70edbaf3e004aaa"}, + {file = "coverage-6.3.2-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e5f4e1edcf57ce94e5475fe09e5afa3e3145081318e5fd1a43a6b4539a97e518"}, + {file = "coverage-6.3.2-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7a15dc0a14008f1da3d1ebd44bdda3e357dbabdf5a0b5034d38fcde0b5c234b7"}, + {file = "coverage-6.3.2-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:21b7745788866028adeb1e0eca3bf1101109e2dc58456cb49d2d9b99a8c516e6"}, + {file = "coverage-6.3.2-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:8ce257cac556cb03be4a248d92ed36904a59a4a5ff55a994e92214cde15c5bad"}, + {file = "coverage-6.3.2-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:b0be84e5a6209858a1d3e8d1806c46214e867ce1b0fd32e4ea03f4bd8b2e3359"}, + {file = "coverage-6.3.2-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:acf53bc2cf7282ab9b8ba346746afe703474004d9e566ad164c91a7a59f188a4"}, + {file = "coverage-6.3.2-cp37-cp37m-win32.whl", hash = "sha256:8bdde1177f2311ee552f47ae6e5aa7750c0e3291ca6b75f71f7ffe1f1dab3dca"}, + {file = "coverage-6.3.2-cp37-cp37m-win_amd64.whl", hash = "sha256:b31651d018b23ec463e95cf10070d0b2c548aa950a03d0b559eaa11c7e5a6fa3"}, + {file = "coverage-6.3.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:07e6db90cd9686c767dcc593dff16c8c09f9814f5e9c51034066cad3373b914d"}, + {file = "coverage-6.3.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:2c6dbb42f3ad25760010c45191e9757e7dce981cbfb90e42feef301d71540059"}, + {file = "coverage-6.3.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c76aeef1b95aff3905fb2ae2d96e319caca5b76fa41d3470b19d4e4a3a313512"}, + {file = "coverage-6.3.2-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:8cf5cfcb1521dc3255d845d9dca3ff204b3229401994ef8d1984b32746bb45ca"}, + {file = "coverage-6.3.2-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8fbbdc8d55990eac1b0919ca69eb5a988a802b854488c34b8f37f3e2025fa90d"}, + {file = "coverage-6.3.2-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:ec6bc7fe73a938933d4178c9b23c4e0568e43e220aef9472c4f6044bfc6dd0f0"}, + {file = "coverage-6.3.2-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:9baff2a45ae1f17c8078452e9e5962e518eab705e50a0aa8083733ea7d45f3a6"}, + {file = "coverage-6.3.2-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:fd9e830e9d8d89b20ab1e5af09b32d33e1a08ef4c4e14411e559556fd788e6b2"}, + {file = "coverage-6.3.2-cp38-cp38-win32.whl", hash = "sha256:f7331dbf301b7289013175087636bbaf5b2405e57259dd2c42fdcc9fcc47325e"}, + {file = "coverage-6.3.2-cp38-cp38-win_amd64.whl", hash = "sha256:68353fe7cdf91f109fc7d474461b46e7f1f14e533e911a2a2cbb8b0fc8613cf1"}, + {file = "coverage-6.3.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:b78e5afb39941572209f71866aa0b206c12f0109835aa0d601e41552f9b3e620"}, + {file = "coverage-6.3.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4e21876082ed887baed0146fe222f861b5815455ada3b33b890f4105d806128d"}, + {file = "coverage-6.3.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:34626a7eee2a3da12af0507780bb51eb52dca0e1751fd1471d0810539cefb536"}, + {file = "coverage-6.3.2-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1ebf730d2381158ecf3dfd4453fbca0613e16eaa547b4170e2450c9707665ce7"}, + {file = "coverage-6.3.2-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:dd6fe30bd519694b356cbfcaca9bd5c1737cddd20778c6a581ae20dc8c04def2"}, + {file = "coverage-6.3.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:96f8a1cb43ca1422f36492bebe63312d396491a9165ed3b9231e778d43a7fca4"}, + {file = "coverage-6.3.2-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:dd035edafefee4d573140a76fdc785dc38829fe5a455c4bb12bac8c20cfc3d69"}, + {file = "coverage-6.3.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5ca5aeb4344b30d0bec47481536b8ba1181d50dbe783b0e4ad03c95dc1296684"}, + {file = "coverage-6.3.2-cp39-cp39-win32.whl", hash = "sha256:f5fa5803f47e095d7ad8443d28b01d48c0359484fec1b9d8606d0e3282084bc4"}, + {file = "coverage-6.3.2-cp39-cp39-win_amd64.whl", hash = "sha256:9548f10d8be799551eb3a9c74bbf2b4934ddb330e08a73320123c07f95cc2d92"}, + {file = "coverage-6.3.2-pp36.pp37.pp38-none-any.whl", hash = "sha256:18d520c6860515a771708937d2f78f63cc47ab3b80cb78e86573b0a760161faf"}, + {file = "coverage-6.3.2.tar.gz", hash = "sha256:03e2a7826086b91ef345ff18742ee9fc47a6839ccd517061ef8fa1976e652ce9"}, ] darglint = [ - {file = "darglint-1.8.0-py3-none-any.whl", hash = "sha256:ac6797bcc918cd8d8f14c168a4a364f54e1aeb4ced59db58e7e4c6dfec2fe15c"}, - {file = "darglint-1.8.0.tar.gz", hash = "sha256:aa605ef47817a6d14797d32b390466edab621768ea4ca5cc0f3c54f6d8dcaec8"}, -] -distlib = [ - {file = "distlib-0.3.2-py2.py3-none-any.whl", hash = "sha256:23e223426b28491b1ced97dc3bbe183027419dfc7982b4fa2f05d5f3ff10711c"}, - {file = "distlib-0.3.2.zip", hash = "sha256:106fef6dc37dd8c0e2c0a60d3fca3e77460a48907f335fa28420463a6f799736"}, + {file = "darglint-1.8.1-py3-none-any.whl", hash = "sha256:5ae11c259c17b0701618a20c3da343a3eb98b3bc4b5a83d31cdd94f5ebdced8d"}, + {file = "darglint-1.8.1.tar.gz", hash = "sha256:080d5106df149b199822e7ee7deb9c012b49891538f14a11be681044f0bb20da"}, ] docutils = [ - {file = "docutils-0.17.1-py2.py3-none-any.whl", hash = "sha256:cf316c8370a737a022b72b56874f6602acf974a37a9fba42ec2876387549fc61"}, - {file = "docutils-0.17.1.tar.gz", hash = "sha256:686577d2e4c32380bb50cbb22f575ed742d58168cee37e99117a854bcd88f125"}, + {file = "docutils-0.18.1-py2.py3-none-any.whl", hash = "sha256:23010f129180089fbcd3bc08cfefccb3b890b0050e1ca00c867036e9d161b98c"}, + {file = "docutils-0.18.1.tar.gz", hash = "sha256:679987caf361a7539d76e584cbeddc311e3aee937877c87346f31debc63e9d06"}, ] eradicate = [ {file = "eradicate-2.0.0.tar.gz", hash = "sha256:27434596f2c5314cc9b31410c93d8f7e8885747399773cd088d3adea647a60c8"}, ] -filelock = [ - {file = "filelock-3.0.12-py3-none-any.whl", hash = "sha256:929b7d63ec5b7d6b71b0fa5ac14e030b3f70b75747cef1b10da9b879fef15836"}, - {file = "filelock-3.0.12.tar.gz", hash = "sha256:18d82244ee114f543149c66a6e0c14e9c4f8a1044b5cdaadd0f82159d6a6ff59"}, +fastapi = [ + {file = "fastapi-0.73.0-py3-none-any.whl", hash = "sha256:f0a618aff5f6942862f2d3f20f39b1c037e33314d1b8207fd1c3a2cca76dfd8c"}, + {file = "fastapi-0.73.0.tar.gz", hash = "sha256:dcfee92a7f9a72b5d4b7ca364bd2b009f8fc10d95ed5769be20e94f39f7e5a15"}, ] flake8 = [ - {file = "flake8-3.9.2-py2.py3-none-any.whl", hash = "sha256:bf8fd333346d844f616e8d47905ef3a3384edae6b4e9beb0c5101e25e3110907"}, - {file = "flake8-3.9.2.tar.gz", hash = "sha256:07528381786f2a6237b061f6e96610a4167b226cb926e2aa2b6b1d78057c576b"}, + {file = "flake8-4.0.1-py2.py3-none-any.whl", hash = "sha256:479b1304f72536a55948cb40a32dce8bb0ffe3501e26eaf292c7e60eb5e0428d"}, + {file = "flake8-4.0.1.tar.gz", hash = "sha256:806e034dda44114815e23c16ef92f95c91e4c71100ff52813adf7132a6ad870d"}, ] flake8-bandit = [ {file = "flake8_bandit-2.1.2.tar.gz", hash = "sha256:687fc8da2e4a239b206af2e54a90093572a60d0954f3054e23690739b0b0de3b"}, ] flake8-broken-line = [ - {file = "flake8-broken-line-0.3.0.tar.gz", hash = "sha256:f74e052833324a9e5f0055032f7ccc54b23faabafe5a26241c2f977e70b10b50"}, - {file = "flake8_broken_line-0.3.0-py3-none-any.whl", hash = "sha256:611f79c7f27118e7e5d3dc098ef7681c40aeadf23783700c5dbee840d2baf3af"}, + {file = "flake8-broken-line-0.4.0.tar.gz", hash = "sha256:771aab5aa0997666796fed249d0e48e6c01cdfeca8c95521eea28a38b7ced4c7"}, + {file = "flake8_broken_line-0.4.0-py3-none-any.whl", hash = "sha256:e9c522856862239a2c7ef2c1de0276fa598572aa864bd4e9c7efc2a827538515"}, ] flake8-bugbear = [ - {file = "flake8-bugbear-21.9.1.tar.gz", hash = "sha256:2f60c8ce0dc53d51da119faab2d67dea978227f0f92ed3c44eb7d65fb2e06a96"}, - {file = "flake8_bugbear-21.9.1-py36.py37.py38-none-any.whl", hash = "sha256:45bfdccfb9f2d8aa140e33cac8f46f1e38215c13d5aa8650e7e188d84e2f94c6"}, + {file = "flake8-bugbear-21.11.29.tar.gz", hash = "sha256:8b04cb2fafc6a78e1a9d873bd3988e4282f7959bb6b0d7c1ae648ec09b937a7b"}, + {file = "flake8_bugbear-21.11.29-py36.py37.py38-none-any.whl", hash = "sha256:179e41ddae5de5e3c20d1f61736feeb234e70958fbb56ab3c28a67739c8e9a82"}, ] flake8-commas = [ - {file = "flake8-commas-2.0.0.tar.gz", hash = "sha256:d3005899466f51380387df7151fb59afec666a0f4f4a2c6a8995b975de0f44b7"}, - {file = "flake8_commas-2.0.0-py2.py3-none-any.whl", hash = "sha256:ee2141a3495ef9789a3894ed8802d03eff1eaaf98ce6d8653a7c573ef101935e"}, + {file = "flake8-commas-2.1.0.tar.gz", hash = "sha256:940441ab8ee544df564ae3b3f49f20462d75d5c7cac2463e0b27436e2050f263"}, + {file = "flake8_commas-2.1.0-py2.py3-none-any.whl", hash = "sha256:ebb96c31e01d0ef1d0685a21f3f0e2f8153a0381430e748bf0bbbb5d5b453d54"}, ] flake8-comprehensions = [ - {file = "flake8-comprehensions-3.6.1.tar.gz", hash = "sha256:4888de89248b7f7535159189ff693c77f8354f6d37a02619fa28c9921a913aa0"}, - {file = "flake8_comprehensions-3.6.1-py3-none-any.whl", hash = "sha256:e9a010b99aa90c05790d45281ad9953df44a4a08a1a8f6cd41f98b4fc6a268a0"}, + {file = "flake8-comprehensions-3.8.0.tar.gz", hash = "sha256:8e108707637b1d13734f38e03435984f6b7854fa6b5a4e34f93e69534be8e521"}, + {file = "flake8_comprehensions-3.8.0-py3-none-any.whl", hash = "sha256:9406314803abe1193c064544ab14fdc43c58424c0882f6ff8a581eb73fc9bb58"}, ] flake8-debugger = [ {file = "flake8-debugger-4.0.0.tar.gz", hash = "sha256:e43dc777f7db1481db473210101ec2df2bd39a45b149d7218a618e954177eda6"}, @@ -1512,148 +1303,121 @@ flake8-docstrings = [ {file = "flake8_docstrings-1.6.0-py2.py3-none-any.whl", hash = "sha256:99cac583d6c7e32dd28bbfbef120a7c0d1b6dde4adb5a9fd441c4227a6534bde"}, ] flake8-eradicate = [ - {file = "flake8-eradicate-1.1.0.tar.gz", hash = "sha256:f5917d6dbca352efcd10c15fdab9c55c48f0f26f6a8d47898b25d39101f170a8"}, - {file = "flake8_eradicate-1.1.0-py3-none-any.whl", hash = "sha256:d8e39b684a37c257a53cda817d86e2d96c9ba3450ddc292742623a5dfee04d9e"}, + {file = "flake8-eradicate-1.2.0.tar.gz", hash = "sha256:acaa1b6839ff00d284b805c432fdfa6047262bd15a5504ec945797e87b4de1fa"}, + {file = "flake8_eradicate-1.2.0-py3-none-any.whl", hash = "sha256:51dc660d0c1c1ed93af0f813540bbbf72ab2d3466c14e3f3bac371c618b6042f"}, ] flake8-isort = [ - {file = "flake8-isort-4.0.0.tar.gz", hash = "sha256:2b91300f4f1926b396c2c90185844eb1a3d5ec39ea6138832d119da0a208f4d9"}, - {file = "flake8_isort-4.0.0-py2.py3-none-any.whl", hash = "sha256:729cd6ef9ba3659512dee337687c05d79c78e1215fdf921ed67e5fe46cce2f3c"}, -] -flake8-plugin-utils = [ - {file = "flake8-plugin-utils-1.3.2.tar.gz", hash = "sha256:20fa2a8ca2decac50116edb42e6af0a1253ef639ad79941249b840531889c65a"}, - {file = "flake8_plugin_utils-1.3.2-py3-none-any.whl", hash = "sha256:1fe43e3e9acf3a7c0f6b88f5338cad37044d2f156c43cb6b080b5f9da8a76f06"}, + {file = "flake8-isort-4.1.1.tar.gz", hash = "sha256:d814304ab70e6e58859bc5c3e221e2e6e71c958e7005239202fee19c24f82717"}, + {file = "flake8_isort-4.1.1-py3-none-any.whl", hash = "sha256:c4e8b6dcb7be9b71a02e6e5d4196cefcef0f3447be51e82730fb336fff164949"}, ] flake8-polyfill = [ {file = "flake8-polyfill-1.0.2.tar.gz", hash = "sha256:e44b087597f6da52ec6393a709e7108b2905317d0c0b744cdca6208e670d8eda"}, {file = "flake8_polyfill-1.0.2-py2.py3-none-any.whl", hash = "sha256:12be6a34ee3ab795b19ca73505e7b55826d5f6ad7230d31b18e106400169b9e9"}, ] -flake8-pytest-style = [ - {file = "flake8-pytest-style-1.5.0.tar.gz", hash = "sha256:668ce8f55edf7db4ac386d2735c3b354b5cb47aa341a4655d91a5788dd03124b"}, - {file = "flake8_pytest_style-1.5.0-py3-none-any.whl", hash = "sha256:ec287a7dc4fe95082af5e408c8b2f8f4b6bcb366d5a17ff6c34112eb03446580"}, -] flake8-quotes = [ - {file = "flake8-quotes-3.3.0.tar.gz", hash = "sha256:f1dd87830ed77ff2ce47fc0ee0fd87ae20e8f045355354ffbf4dcaa18d528217"}, + {file = "flake8-quotes-3.3.1.tar.gz", hash = "sha256:633adca6fb8a08131536af0d750b44d6985b9aba46f498871e21588c3e6f525a"}, ] flake8-rst-docstrings = [ - {file = "flake8-rst-docstrings-0.2.3.tar.gz", hash = "sha256:3045794e1c8467fba33aaea5c246b8369efc9c44ef8b0b20199bb6df7a4bd47b"}, - {file = "flake8_rst_docstrings-0.2.3-py3-none-any.whl", hash = "sha256:565bbb391d7e4d0042924102221e9857ad72929cdd305b26501736ec22c1451a"}, + {file = "flake8-rst-docstrings-0.2.5.tar.gz", hash = "sha256:4fe93f997dea45d9d3c8bd220f12f0b6c359948fb943b5b48021a3f927edd816"}, + {file = "flake8_rst_docstrings-0.2.5-py3-none-any.whl", hash = "sha256:b99d9041b769b857efe45a448dc8c71b1bb311f9cacbdac5de82f96498105082"}, ] flake8-string-format = [ {file = "flake8-string-format-0.3.0.tar.gz", hash = "sha256:65f3da786a1461ef77fca3780b314edb2853c377f2e35069723348c8917deaa2"}, {file = "flake8_string_format-0.3.0-py2.py3-none-any.whl", hash = "sha256:812ff431f10576a74c89be4e85b8e075a705be39bc40c4b4278b5b13e2afa9af"}, ] ghp-import = [ - {file = "ghp-import-2.0.1.tar.gz", hash = "sha256:753de2eace6e0f7d4edfb3cce5e3c3b98cd52aadb80163303d1d036bda7b4483"}, + {file = "ghp-import-2.0.2.tar.gz", hash = "sha256:947b3771f11be850c852c64b561c600fdddf794bab363060854c1ee7ad05e071"}, + {file = "ghp_import-2.0.2-py3-none-any.whl", hash = "sha256:5f8962b30b20652cdffa9c5a9812f7de6bcb56ec475acac579807719bf242c46"}, ] gitdb = [ - {file = "gitdb-4.0.7-py3-none-any.whl", hash = "sha256:6c4cc71933456991da20917998acbe6cf4fb41eeaab7d6d67fbc05ecd4c865b0"}, - {file = "gitdb-4.0.7.tar.gz", hash = "sha256:96bf5c08b157a666fec41129e6d327235284cca4c81e92109260f353ba138005"}, + {file = "gitdb-4.0.9-py3-none-any.whl", hash = "sha256:8033ad4e853066ba6ca92050b9df2f89301b8fc8bf7e9324d412a63f8bf1a8fd"}, + {file = "gitdb-4.0.9.tar.gz", hash = "sha256:bac2fd45c0a1c9cf619e63a90d62bdc63892ef92387424b855792a6cabe789aa"}, ] gitpython = [ - {file = "GitPython-3.1.23-py3-none-any.whl", hash = "sha256:de2e2aff068097b23d6dca5daf588078fd8996a4218f6ffa704a662c2b54f9ac"}, - {file = "GitPython-3.1.23.tar.gz", hash = "sha256:aaae7a3bfdf0a6db30dc1f3aeae47b71cd326d86b936fe2e158aa925fdf1471c"}, + {file = "GitPython-3.1.27-py3-none-any.whl", hash = "sha256:5b68b000463593e05ff2b261acff0ff0972df8ab1b70d3cdbd41b546c8b8fc3d"}, + {file = "GitPython-3.1.27.tar.gz", hash = "sha256:1c885ce809e8ba2d88a29befeb385fcea06338d3640712b59ca623c220bb5704"}, ] h11 = [ {file = "h11-0.12.0-py3-none-any.whl", hash = "sha256:36a3cb8c0a032f56e2da7084577878a035d3b61d104230d4bd49c0c6b555a9c6"}, {file = "h11-0.12.0.tar.gz", hash = "sha256:47222cb6067e4a307d535814917cd98fd0a57b6788ce715755fa2b6c28b56042"}, ] httpcore = [ - {file = "httpcore-0.13.7-py3-none-any.whl", hash = "sha256:369aa481b014cf046f7067fddd67d00560f2f00426e79569d99cb11245134af0"}, - {file = "httpcore-0.13.7.tar.gz", hash = "sha256:036f960468759e633574d7c121afba48af6419615d36ab8ede979f1ad6276fa3"}, + {file = "httpcore-0.14.7-py3-none-any.whl", hash = "sha256:47d772f754359e56dd9d892d9593b6f9870a37aeb8ba51e9a88b09b3d68cfade"}, + {file = "httpcore-0.14.7.tar.gz", hash = "sha256:7503ec1c0f559066e7e39bc4003fd2ce023d01cf51793e3c173b864eb456ead1"}, ] httpx = [ - {file = "httpx-0.19.0-py3-none-any.whl", hash = "sha256:9bd728a6c5ec0a9e243932a9983d57d3cc4a87bb4f554e1360fce407f78f9435"}, - {file = "httpx-0.19.0.tar.gz", hash = "sha256:92ecd2c00c688b529eda11cedb15161eaf02dee9116712f621c70d9a40b2cdd0"}, + {file = "httpx-0.21.3-py3-none-any.whl", hash = "sha256:df9a0fd43fa79dbab411d83eb1ea6f7a525c96ad92e60c2d7f40388971b25777"}, + {file = "httpx-0.21.3.tar.gz", hash = "sha256:7a3eb67ef0b8abbd6d9402248ef2f84a76080fa1c839f8662e6eb385640e445a"}, ] idna = [ - {file = "idna-3.2-py3-none-any.whl", hash = "sha256:14475042e284991034cb48e06f6851428fb14c4dc953acd9be9a5e95c7b6dd7a"}, - {file = "idna-3.2.tar.gz", hash = "sha256:467fbad99067910785144ce333826c71fb0e63a425657295239737f7ecd125f3"}, + {file = "idna-3.3-py3-none-any.whl", hash = "sha256:84d9dd047ffa80596e0f246e2eab0b391788b0503584e8945f2368256d2735ff"}, + {file = "idna-3.3.tar.gz", hash = "sha256:9d643ff0a55b762d5cdb124b8eaa99c66322e2157b69160bc32796e824360e6d"}, ] importlib-metadata = [ - {file = "importlib_metadata-4.8.1-py3-none-any.whl", hash = "sha256:b618b6d2d5ffa2f16add5697cf57a46c76a56229b0ed1c438322e4e95645bd15"}, - {file = "importlib_metadata-4.8.1.tar.gz", hash = "sha256:f284b3e11256ad1e5d03ab86bb2ccd6f5339688ff17a4d797a0fe7df326f23b1"}, + {file = "importlib_metadata-4.11.3-py3-none-any.whl", hash = "sha256:1208431ca90a8cca1a6b8af391bb53c1a2db74e5d1cef6ddced95d4b2062edc6"}, + {file = "importlib_metadata-4.11.3.tar.gz", hash = "sha256:ea4c597ebf37142f827b8f39299579e31685c31d3a438b59f469406afd0f2539"}, ] iniconfig = [ {file = "iniconfig-1.1.1-py2.py3-none-any.whl", hash = "sha256:011e24c64b7f47f6ebd835bb12a743f2fbe9a26d4cecaa7f53bc4f35ee9da8b3"}, {file = "iniconfig-1.1.1.tar.gz", hash = "sha256:bc3af051d7d14b2ee5ef9969666def0cd1a000e121eaea580d4a313df4b37f32"}, ] isort = [ - {file = "isort-5.9.3-py3-none-any.whl", hash = "sha256:e17d6e2b81095c9db0a03a8025a957f334d6ea30b26f9ec70805411e5c7c81f2"}, - {file = "isort-5.9.3.tar.gz", hash = "sha256:9c2ea1e62d871267b78307fe511c0838ba0da28698c5732d54e2790bf3ba9899"}, + {file = "isort-5.10.1-py3-none-any.whl", hash = "sha256:6f62d78e2f89b4500b080fe3a81690850cd254227f27f75c3a0c491a1f351ba7"}, + {file = "isort-5.10.1.tar.gz", hash = "sha256:e8443a5e7a020e9d7f97f1d7d9cd17c88bcb3bc7e218bf9cf5095fe550be2951"}, ] jinja2 = [ - {file = "Jinja2-3.0.1-py3-none-any.whl", hash = "sha256:1f06f2da51e7b56b8f238affdd6b4e2c61e39598a378cc49345bc1bd42a978a4"}, - {file = "Jinja2-3.0.1.tar.gz", hash = "sha256:703f484b47a6af502e743c9122595cc812b0271f661722403114f71a79d0f5a4"}, -] -livereload = [ - {file = "livereload-2.6.3.tar.gz", hash = "sha256:776f2f865e59fde56490a56bcc6773b6917366bce0c267c60ee8aaf1a0959869"}, + {file = "Jinja2-3.0.3-py3-none-any.whl", hash = "sha256:077ce6014f7b40d03b47d1f1ca4b0fc8328a692bd284016f806ed0eaca390ad8"}, + {file = "Jinja2-3.0.3.tar.gz", hash = "sha256:611bb273cd68f3b993fabdc4064fc858c5b47a973cb5aa7999ec1ba405c87cd7"}, ] loguru = [ - {file = "loguru-0.5.3-py3-none-any.whl", hash = "sha256:f8087ac396b5ee5f67c963b495d615ebbceac2796379599820e324419d53667c"}, - {file = "loguru-0.5.3.tar.gz", hash = "sha256:b28e72ac7a98be3d28ad28570299a393dfcd32e5e3f6a353dec94675767b6319"}, + {file = "loguru-0.6.0-py3-none-any.whl", hash = "sha256:4e2414d534a2ab57573365b3e6d0234dfb1d84b68b7f3b948e6fb743860a77c3"}, + {file = "loguru-0.6.0.tar.gz", hash = "sha256:066bd06758d0a513e9836fd9c6b5a75bfb3fd36841f4b996bc60b547a309d41c"}, ] markdown = [ - {file = "Markdown-3.3.4-py3-none-any.whl", hash = "sha256:96c3ba1261de2f7547b46a00ea8463832c921d3f9d6aba3f255a6f71386db20c"}, - {file = "Markdown-3.3.4.tar.gz", hash = "sha256:31b5b491868dcc87d6c24b7e3d19a0d730d59d3e46f4eea6430a321bed387a49"}, -] -markdown-include = [ - {file = "markdown-include-0.6.0.tar.gz", hash = "sha256:6f5d680e36f7780c7f0f61dca53ca581bd50d1b56137ddcd6353efafa0c3e4a2"}, + {file = "Markdown-3.3.6-py3-none-any.whl", hash = "sha256:9923332318f843411e9932237530df53162e29dc7a4e2b91e35764583c46c9a3"}, + {file = "Markdown-3.3.6.tar.gz", hash = "sha256:76df8ae32294ec39dcf89340382882dfa12975f87f45c3ed1ecdb1e8cefc7006"}, ] markupsafe = [ - {file = "MarkupSafe-2.0.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:d8446c54dc28c01e5a2dbac5a25f071f6653e6e40f3a8818e8b45d790fe6ef53"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:36bc903cbb393720fad60fc28c10de6acf10dc6cc883f3e24ee4012371399a38"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2d7d807855b419fc2ed3e631034685db6079889a1f01d5d9dac950f764da3dad"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:add36cb2dbb8b736611303cd3bfcee00afd96471b09cda130da3581cbdc56a6d"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:168cd0a3642de83558a5153c8bd34f175a9a6e7f6dc6384b9655d2697312a646"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-win32.whl", hash = "sha256:99df47edb6bda1249d3e80fdabb1dab8c08ef3975f69aed437cb69d0a5de1e28"}, - {file = "MarkupSafe-2.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:e0f138900af21926a02425cf736db95be9f4af72ba1bb21453432a07f6082134"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:0955295dd5eec6cb6cc2fe1698f4c6d84af2e92de33fbcac4111913cd100a6ff"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:0446679737af14f45767963a1a9ef7620189912317d095f2d9ffa183a4d25d2b"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:f826e31d18b516f653fe296d967d700fddad5901ae07c622bb3705955e1faa94"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:fa130dd50c57d53368c9d59395cb5526eda596d3ffe36666cd81a44d56e48872"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:905fec760bd2fa1388bb5b489ee8ee5f7291d692638ea5f67982d968366bef9f"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf5d821ffabf0ef3533c39c518f3357b171a1651c1ff6827325e4489b0e46c3c"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0d4b31cc67ab36e3392bbf3862cfbadac3db12bdd8b02a2731f509ed5b829724"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:baa1a4e8f868845af802979fcdbf0bb11f94f1cb7ced4c4b8a351bb60d108145"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-win32.whl", hash = "sha256:6c4ca60fa24e85fe25b912b01e62cb969d69a23a5d5867682dd3e80b5b02581d"}, - {file = "MarkupSafe-2.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b2f4bf27480f5e5e8ce285a8c8fd176c0b03e93dcc6646477d4630e83440c6a9"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0717a7390a68be14b8c793ba258e075c6f4ca819f15edfc2a3a027c823718567"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:6557b31b5e2c9ddf0de32a691f2312a32f77cd7681d8af66c2692efdbef84c18"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:49e3ceeabbfb9d66c3aef5af3a60cc43b85c33df25ce03d0031a608b0a8b2e3f"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:d7f9850398e85aba693bb640262d3611788b1f29a79f0c93c565694658f4071f"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:6a7fae0dd14cf60ad5ff42baa2e95727c3d81ded453457771d02b7d2b3f9c0c2"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:b7f2d075102dc8c794cbde1947378051c4e5180d52d276987b8d28a3bd58c17d"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e9936f0b261d4df76ad22f8fee3ae83b60d7c3e871292cd42f40b81b70afae85"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:2a7d351cbd8cfeb19ca00de495e224dea7e7d919659c2841bbb7f420ad03e2d6"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:60bf42e36abfaf9aff1f50f52644b336d4f0a3fd6d8a60ca0d054ac9f713a864"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-win32.whl", hash = "sha256:a30e67a65b53ea0a5e62fe23682cfe22712e01f453b95233b25502f7c61cb415"}, - {file = "MarkupSafe-2.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:611d1ad9a4288cf3e3c16014564df047fe08410e628f89805e475368bd304914"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5bb28c636d87e840583ee3adeb78172efc47c8b26127267f54a9c0ec251d41a9"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:be98f628055368795d818ebf93da628541e10b75b41c559fdf36d104c5787066"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:1d609f577dc6e1aa17d746f8bd3c31aa4d258f4070d61b2aa5c4166c1539de35"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7d91275b0245b1da4d4cfa07e0faedd5b0812efc15b702576d103293e252af1b"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:01a9b8ea66f1658938f65b93a85ebe8bc016e6769611be228d797c9d998dd298"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:47ab1e7b91c098ab893b828deafa1203de86d0bc6ab587b160f78fe6c4011f75"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:97383d78eb34da7e1fa37dd273c20ad4320929af65d156e35a5e2d89566d9dfb"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fcf051089389abe060c9cd7caa212c707e58153afa2c649f00346ce6d260f1b"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:5855f8438a7d1d458206a2466bf82b0f104a3724bf96a1c781ab731e4201731a"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:3dd007d54ee88b46be476e293f48c85048603f5f516008bee124ddd891398ed6"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-win32.whl", hash = "sha256:023cb26ec21ece8dc3907c0e8320058b2e0cb3c55cf9564da612bc325bed5e64"}, - {file = "MarkupSafe-2.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:984d76483eb32f1bcb536dc27e4ad56bba4baa70be32fa87152832cdd9db0833"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:2ef54abee730b502252bcdf31b10dacb0a416229b72c18b19e24a4509f273d26"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:3c112550557578c26af18a1ccc9e090bfe03832ae994343cfdacd287db6a6ae7"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:53edb4da6925ad13c07b6d26c2a852bd81e364f95301c66e930ab2aef5b5ddd8"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:f5653a225f31e113b152e56f154ccbe59eeb1c7487b39b9d9f9cdb58e6c79dc5"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:4efca8f86c54b22348a5467704e3fec767b2db12fc39c6d963168ab1d3fc9135"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:ab3ef638ace319fa26553db0624c4699e31a28bb2a835c5faca8f8acf6a5a902"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:f8ba0e8349a38d3001fae7eadded3f6606f0da5d748ee53cc1dab1d6527b9509"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c47adbc92fc1bb2b3274c4b3a43ae0e4573d9fbff4f54cd484555edbf030baf1"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:37205cac2a79194e3750b0af2a5720d95f786a55ce7df90c3af697bfa100eaac"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1f2ade76b9903f39aa442b4aadd2177decb66525062db244b35d71d0ee8599b6"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-win32.whl", hash = "sha256:10f82115e21dc0dfec9ab5c0223652f7197feb168c940f3ef61563fc2d6beb74"}, - {file = "MarkupSafe-2.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:693ce3f9e70a6cf7d2fb9e6c9d8b204b6b39897a2c4a1aa65728d5ac97dcc1d8"}, - {file = "MarkupSafe-2.0.1.tar.gz", hash = "sha256:594c67807fb16238b30c44bdf74f36c02cdf22d1c8cda91ef8a0ed8dabf5620a"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:86b1f75c4e7c2ac2ccdaec2b9022845dbb81880ca318bb7a0a01fbf7813e3812"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:f121a1420d4e173a5d96e47e9a0c0dcff965afdf1626d28de1460815f7c4ee7a"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a49907dd8420c5685cfa064a1335b6754b74541bbb3706c259c02ed65b644b3e"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:10c1bfff05d95783da83491be968e8fe789263689c02724e0c691933c52994f5"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:b7bd98b796e2b6553da7225aeb61f447f80a1ca64f41d83612e6139ca5213aa4"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:b09bf97215625a311f669476f44b8b318b075847b49316d3e28c08e41a7a573f"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:694deca8d702d5db21ec83983ce0bb4b26a578e71fbdbd4fdcd387daa90e4d5e"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:efc1913fd2ca4f334418481c7e595c00aad186563bbc1ec76067848c7ca0a933"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-win32.whl", hash = "sha256:4a33dea2b688b3190ee12bd7cfa29d39c9ed176bda40bfa11099a3ce5d3a7ac6"}, + {file = "MarkupSafe-2.1.1-cp310-cp310-win_amd64.whl", hash = "sha256:dda30ba7e87fbbb7eab1ec9f58678558fd9a6b8b853530e176eabd064da81417"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:671cd1187ed5e62818414afe79ed29da836dde67166a9fac6d435873c44fdd02"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3799351e2336dc91ea70b034983ee71cf2f9533cdff7c14c90ea126bfd95d65a"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e72591e9ecd94d7feb70c1cbd7be7b3ebea3f548870aa91e2732960fa4d57a37"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:6fbf47b5d3728c6aea2abb0589b5d30459e369baa772e0f37a0320185e87c980"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d5ee4f386140395a2c818d149221149c54849dfcfcb9f1debfe07a8b8bd63f9a"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:bcb3ed405ed3222f9904899563d6fc492ff75cce56cba05e32eff40e6acbeaa3"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:e1c0b87e09fa55a220f058d1d49d3fb8df88fbfab58558f1198e08c1e1de842a"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-win32.whl", hash = "sha256:8dc1c72a69aa7e082593c4a203dcf94ddb74bb5c8a731e4e1eb68d031e8498ff"}, + {file = "MarkupSafe-2.1.1-cp37-cp37m-win_amd64.whl", hash = "sha256:97a68e6ada378df82bc9f16b800ab77cbf4b2fada0081794318520138c088e4a"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:e8c843bbcda3a2f1e3c2ab25913c80a3c5376cd00c6e8c4a86a89a28c8dc5452"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0212a68688482dc52b2d45013df70d169f542b7394fc744c02a57374a4207003"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8e576a51ad59e4bfaac456023a78f6b5e6e7651dcd383bcc3e18d06f9b55d6d1"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4b9fe39a2ccc108a4accc2676e77da025ce383c108593d65cc909add5c3bd601"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:96e37a3dc86e80bf81758c152fe66dbf60ed5eca3d26305edf01892257049925"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:6d0072fea50feec76a4c418096652f2c3238eaa014b2f94aeb1d56a66b41403f"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:089cf3dbf0cd6c100f02945abeb18484bd1ee57a079aefd52cffd17fba910b88"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:6a074d34ee7a5ce3effbc526b7083ec9731bb3cbf921bbe1d3005d4d2bdb3a63"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-win32.whl", hash = "sha256:421be9fbf0ffe9ffd7a378aafebbf6f4602d564d34be190fc19a193232fd12b1"}, + {file = "MarkupSafe-2.1.1-cp38-cp38-win_amd64.whl", hash = "sha256:fc7b548b17d238737688817ab67deebb30e8073c95749d55538ed473130ec0c7"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:e04e26803c9c3851c931eac40c695602c6295b8d432cbe78609649ad9bd2da8a"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:b87db4360013327109564f0e591bd2a3b318547bcef31b468a92ee504d07ae4f"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:99a2a507ed3ac881b975a2976d59f38c19386d128e7a9a18b7df6fff1fd4c1d6"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:56442863ed2b06d19c37f94d999035e15ee982988920e12a5b4ba29b62ad1f77"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:3ce11ee3f23f79dbd06fb3d63e2f6af7b12db1d46932fe7bd8afa259a5996603"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:33b74d289bd2f5e527beadcaa3f401e0df0a89927c1559c8566c066fa4248ab7"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:43093fb83d8343aac0b1baa75516da6092f58f41200907ef92448ecab8825135"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:8e3dcf21f367459434c18e71b2a9532d96547aef8a871872a5bd69a715c15f96"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-win32.whl", hash = "sha256:d4306c36ca495956b6d568d276ac11fdd9c30a36f1b6eb928070dc5360b22e1c"}, + {file = "MarkupSafe-2.1.1-cp39-cp39-win_amd64.whl", hash = "sha256:46d00d6cfecdde84d40e572d63735ef81423ad31184100411e6e3388d405e247"}, + {file = "MarkupSafe-2.1.1.tar.gz", hash = "sha256:7f91197cc9e48f989d12e4e6fbc46495c446636dfc81b9ccf50bb0ec74b91d4b"}, ] mccabe = [ {file = "mccabe-0.6.1-py2.py3-none-any.whl", hash = "sha256:ab8a6258860da4b6677da4bd2fe5dc2c659cff31b3ee4f7f5d64e79735b80d42"}, @@ -1664,92 +1428,77 @@ mergedeep = [ {file = "mergedeep-1.3.4.tar.gz", hash = "sha256:0096d52e9dad9939c3d975a774666af186eda617e6ca84df4c94dec30004f2a8"}, ] mkdocs = [ - {file = "mkdocs-1.2.2-py3-none-any.whl", hash = "sha256:d019ff8e17ec746afeb54eb9eb4112b5e959597aebc971da46a5c9486137f0ff"}, - {file = "mkdocs-1.2.2.tar.gz", hash = "sha256:a334f5bd98ec960638511366eb8c5abc9c99b9083a0ed2401d8791b112d6b078"}, -] -mkdocs-autorefs = [ - {file = "mkdocs-autorefs-0.2.1.tar.gz", hash = "sha256:b8156d653ed91356e71675ce1fa1186d2b2c2085050012522895c9aa98fca3e5"}, - {file = "mkdocs_autorefs-0.2.1-py3-none-any.whl", hash = "sha256:f301b983a34259df90b3fcf7edc234b5e6c7065bd578781e66fd90b8cfbe76be"}, + {file = "mkdocs-1.2.3-py3-none-any.whl", hash = "sha256:a1fa8c2d0c1305d7fc2b9d9f607c71778572a8b110fb26642aa00296c9e6d072"}, + {file = "mkdocs-1.2.3.tar.gz", hash = "sha256:89f5a094764381cda656af4298727c9f53dc3e602983087e1fe96ea1df24f4c1"}, ] mkdocs-material = [ - {file = "mkdocs-material-7.2.6.tar.gz", hash = "sha256:4bdeff63904680865676ceb3193216934de0b33fa5b2446e0a84ade60929ee54"}, - {file = "mkdocs_material-7.2.6-py2.py3-none-any.whl", hash = "sha256:4c6939b9d7d5c6db948ab02df8525c64211828ddf33286acea8b9d2115cec369"}, + {file = "mkdocs-material-8.1.9.tar.gz", hash = "sha256:a15873a5e116bf4615af4fcedc85a0537492464365286cba50310d96fb066958"}, + {file = "mkdocs_material-8.1.9-py2.py3-none-any.whl", hash = "sha256:6feb433f29227b862418bd1009edeec2e52870770c476bf02840fc094b8823f2"}, ] mkdocs-material-extensions = [ {file = "mkdocs-material-extensions-1.0.3.tar.gz", hash = "sha256:bfd24dfdef7b41c312ede42648f9eb83476ea168ec163b613f9abd12bbfddba2"}, {file = "mkdocs_material_extensions-1.0.3-py3-none-any.whl", hash = "sha256:a82b70e533ce060b2a5d9eb2bc2e1be201cf61f901f93704b4acf6e3d5983a44"}, ] -mkdocstrings = [ - {file = "mkdocstrings-0.15.2-py3-none-any.whl", hash = "sha256:8d6cbe64c07ae66739010979ca01d49dd2f64d1a45009f089d217b9cd2a65e36"}, - {file = "mkdocstrings-0.15.2.tar.gz", hash = "sha256:c2fee9a3a644647c06eb2044fdfede1073adfd1a55bf6752005d3db10705fe73"}, -] -molten = [ - {file = "molten-1.0.2-py3-none-any.whl", hash = "sha256:e4316ecd97e721ac573ff744803542c0ebeee3f3d37575f42614db84f7f6b737"}, - {file = "molten-1.0.2.tar.gz", hash = "sha256:bc119f0f59f2ac1fab447ce23dc44d0d598784f27482c3be8c4fdf3ad722aae1"}, -] mypy = [ - {file = "mypy-0.812-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:a26f8ec704e5a7423c8824d425086705e381b4f1dfdef6e3a1edab7ba174ec49"}, - {file = "mypy-0.812-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:28fb5479c494b1bab244620685e2eb3c3f988d71fd5d64cc753195e8ed53df7c"}, - {file = "mypy-0.812-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:9743c91088d396c1a5a3c9978354b61b0382b4e3c440ce83cf77994a43e8c521"}, - {file = "mypy-0.812-cp35-cp35m-win_amd64.whl", hash = "sha256:d7da2e1d5f558c37d6e8c1246f1aec1e7349e4913d8fb3cb289a35de573fe2eb"}, - {file = "mypy-0.812-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:4eec37370483331d13514c3f55f446fc5248d6373e7029a29ecb7b7494851e7a"}, - {file = "mypy-0.812-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d65cc1df038ef55a99e617431f0553cd77763869eebdf9042403e16089fe746c"}, - {file = "mypy-0.812-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:61a3d5b97955422964be6b3baf05ff2ce7f26f52c85dd88db11d5e03e146a3a6"}, - {file = "mypy-0.812-cp36-cp36m-win_amd64.whl", hash = "sha256:25adde9b862f8f9aac9d2d11971f226bd4c8fbaa89fb76bdadb267ef22d10064"}, - {file = "mypy-0.812-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:552a815579aa1e995f39fd05dde6cd378e191b063f031f2acfe73ce9fb7f9e56"}, - {file = "mypy-0.812-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:499c798053cdebcaa916eef8cd733e5584b5909f789de856b482cd7d069bdad8"}, - {file = "mypy-0.812-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:5873888fff1c7cf5b71efbe80e0e73153fe9212fafdf8e44adfe4c20ec9f82d7"}, - {file = "mypy-0.812-cp37-cp37m-win_amd64.whl", hash = "sha256:9f94aac67a2045ec719ffe6111df543bac7874cee01f41928f6969756e030564"}, - {file = "mypy-0.812-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d23e0ea196702d918b60c8288561e722bf437d82cb7ef2edcd98cfa38905d506"}, - {file = "mypy-0.812-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:674e822aa665b9fd75130c6c5f5ed9564a38c6cea6a6432ce47eafb68ee578c5"}, - {file = "mypy-0.812-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:abf7e0c3cf117c44d9285cc6128856106183938c68fd4944763003decdcfeb66"}, - {file = "mypy-0.812-cp38-cp38-win_amd64.whl", hash = "sha256:0d0a87c0e7e3a9becdfbe936c981d32e5ee0ccda3e0f07e1ef2c3d1a817cf73e"}, - {file = "mypy-0.812-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:7ce3175801d0ae5fdfa79b4f0cfed08807af4d075b402b7e294e6aa72af9aa2a"}, - {file = "mypy-0.812-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:b09669bcda124e83708f34a94606e01b614fa71931d356c1f1a5297ba11f110a"}, - {file = "mypy-0.812-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:33f159443db0829d16f0a8d83d94df3109bb6dd801975fe86bacb9bf71628e97"}, - {file = "mypy-0.812-cp39-cp39-win_amd64.whl", hash = "sha256:3f2aca7f68580dc2508289c729bd49ee929a436208d2b2b6aab15745a70a57df"}, - {file = "mypy-0.812-py3-none-any.whl", hash = "sha256:2f9b3407c58347a452fc0736861593e105139b905cca7d097e413453a1d650b4"}, - {file = "mypy-0.812.tar.gz", hash = "sha256:cd07039aa5df222037005b08fbbfd69b3ab0b0bd7a07d7906de75ae52c4e3119"}, + {file = "mypy-0.910-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:a155d80ea6cee511a3694b108c4494a39f42de11ee4e61e72bc424c490e46457"}, + {file = "mypy-0.910-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:b94e4b785e304a04ea0828759172a15add27088520dc7e49ceade7834275bedb"}, + {file = "mypy-0.910-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:088cd9c7904b4ad80bec811053272986611b84221835e079be5bcad029e79dd9"}, + {file = "mypy-0.910-cp35-cp35m-win_amd64.whl", hash = "sha256:adaeee09bfde366d2c13fe6093a7df5df83c9a2ba98638c7d76b010694db760e"}, + {file = "mypy-0.910-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:ecd2c3fe726758037234c93df7e98deb257fd15c24c9180dacf1ef829da5f921"}, + {file = "mypy-0.910-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:d9dd839eb0dc1bbe866a288ba3c1afc33a202015d2ad83b31e875b5905a079b6"}, + {file = "mypy-0.910-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:3e382b29f8e0ccf19a2df2b29a167591245df90c0b5a2542249873b5c1d78212"}, + {file = "mypy-0.910-cp36-cp36m-win_amd64.whl", hash = "sha256:53fd2eb27a8ee2892614370896956af2ff61254c275aaee4c230ae771cadd885"}, + {file = "mypy-0.910-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b6fb13123aeef4a3abbcfd7e71773ff3ff1526a7d3dc538f3929a49b42be03f0"}, + {file = "mypy-0.910-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e4dab234478e3bd3ce83bac4193b2ecd9cf94e720ddd95ce69840273bf44f6de"}, + {file = "mypy-0.910-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:7df1ead20c81371ccd6091fa3e2878559b5c4d4caadaf1a484cf88d93ca06703"}, + {file = "mypy-0.910-cp37-cp37m-win_amd64.whl", hash = "sha256:0aadfb2d3935988ec3815952e44058a3100499f5be5b28c34ac9d79f002a4a9a"}, + {file = "mypy-0.910-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ec4e0cd079db280b6bdabdc807047ff3e199f334050db5cbb91ba3e959a67504"}, + {file = "mypy-0.910-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:119bed3832d961f3a880787bf621634ba042cb8dc850a7429f643508eeac97b9"}, + {file = "mypy-0.910-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:866c41f28cee548475f146aa4d39a51cf3b6a84246969f3759cb3e9c742fc072"}, + {file = "mypy-0.910-cp38-cp38-win_amd64.whl", hash = "sha256:ceb6e0a6e27fb364fb3853389607cf7eb3a126ad335790fa1e14ed02fba50811"}, + {file = "mypy-0.910-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:1a85e280d4d217150ce8cb1a6dddffd14e753a4e0c3cf90baabb32cefa41b59e"}, + {file = "mypy-0.910-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:42c266ced41b65ed40a282c575705325fa7991af370036d3f134518336636f5b"}, + {file = "mypy-0.910-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:3c4b8ca36877fc75339253721f69603a9c7fdb5d4d5a95a1a1b899d8b86a4de2"}, + {file = "mypy-0.910-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:c0df2d30ed496a08de5daed2a9ea807d07c21ae0ab23acf541ab88c24b26ab97"}, + {file = "mypy-0.910-cp39-cp39-win_amd64.whl", hash = "sha256:c6c2602dffb74867498f86e6129fd52a2770c48b7cd3ece77ada4fa38f94eba8"}, + {file = "mypy-0.910-py3-none-any.whl", hash = "sha256:ef565033fa5a958e62796867b1df10c40263ea9ded87164d67572834e57a174d"}, + {file = "mypy-0.910.tar.gz", hash = "sha256:704098302473cb31a218f1775a873b376b30b4c18229421e9e9dc8916fd16150"}, ] mypy-extensions = [ {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, ] -nox = [ - {file = "nox-2021.6.12-py3-none-any.whl", hash = "sha256:1e90df301f6622efb1c29d1586e5a5755846b1eb99b2764230304f8fa31d1734"}, - {file = "nox-2021.6.12.tar.gz", hash = "sha256:955dbeb8e657a08226f8c1c8f8d1e2a40fe5438a792056314f351e504639a80f"}, -] packaging = [ - {file = "packaging-21.0-py3-none-any.whl", hash = "sha256:c86254f9220d55e31cc94d69bade760f0847da8000def4dfe1c6b872fd14ff14"}, - {file = "packaging-21.0.tar.gz", hash = "sha256:7dc96269f53a4ccec5c0670940a4281106dd0bb343f47b7471f779df49c2fbe7"}, + {file = "packaging-21.3-py3-none-any.whl", hash = "sha256:ef103e05f519cdc783ae24ea4e2e0f508a9c99b2d4969652eed6a2e1ea5bd522"}, + {file = "packaging-21.3.tar.gz", hash = "sha256:dd47c42927d89ab911e606518907cc2d3a1f38bbd026385970643f9c5b8ecfeb"}, ] pathspec = [ {file = "pathspec-0.9.0-py2.py3-none-any.whl", hash = "sha256:7d15c4ddb0b5c802d161efc417ec1a2558ea2653c2e8ad9c19098201dc1c993a"}, {file = "pathspec-0.9.0.tar.gz", hash = "sha256:e564499435a2673d586f6b2130bb5b95f04a3ba06f81b8f895b651a3c76aabb1"}, ] pbr = [ - {file = "pbr-5.6.0-py2.py3-none-any.whl", hash = "sha256:c68c661ac5cc81058ac94247278eeda6d2e6aecb3e227b0387c30d277e7ef8d4"}, - {file = "pbr-5.6.0.tar.gz", hash = "sha256:42df03e7797b796625b1029c0400279c7c34fd7df24a7d7818a1abb5b38710dd"}, + {file = "pbr-5.8.1-py2.py3-none-any.whl", hash = "sha256:27108648368782d07bbf1cb468ad2e2eeef29086affd14087a6d04b7de8af4ec"}, + {file = "pbr-5.8.1.tar.gz", hash = "sha256:66bc5a34912f408bb3925bf21231cb6f59206267b7f63f3503ef865c1a292e25"}, ] pep8-naming = [ - {file = "pep8-naming-0.11.1.tar.gz", hash = "sha256:a1dd47dd243adfe8a83616e27cf03164960b507530f155db94e10b36a6cd6724"}, - {file = "pep8_naming-0.11.1-py2.py3-none-any.whl", hash = "sha256:f43bfe3eea7e0d73e8b5d07d6407ab47f2476ccaeff6937c84275cd30b016738"}, + {file = "pep8-naming-0.12.1.tar.gz", hash = "sha256:bb2455947757d162aa4cad55dba4ce029005cd1692f2899a21d51d8630ca7841"}, + {file = "pep8_naming-0.12.1-py2.py3-none-any.whl", hash = "sha256:4a8daeaeb33cfcde779309fc0c9c0a68a3bbe2ad8a8308b763c5068f86eb9f37"}, ] platformdirs = [ - {file = "platformdirs-2.3.0-py3-none-any.whl", hash = "sha256:8003ac87717ae2c7ee1ea5a84a1a61e87f3fbd16eb5aadba194ea30a9019f648"}, - {file = "platformdirs-2.3.0.tar.gz", hash = "sha256:15b056538719b1c94bdaccb29e5f81879c7f7f0f4a153f46086d155dffcd4f0f"}, + {file = "platformdirs-2.5.1-py3-none-any.whl", hash = "sha256:bcae7cab893c2d310a711b70b24efb93334febe65f8de776ee320b517471e227"}, + {file = "platformdirs-2.5.1.tar.gz", hash = "sha256:7535e70dfa32e84d4b34996ea99c5e432fa29a708d0f4e394bbcb2a8faa4f16d"}, ] pluggy = [ - {file = "pluggy-0.13.1-py2.py3-none-any.whl", hash = "sha256:966c145cd83c96502c3c3868f50408687b38434af77734af1e9ca461a4081d2d"}, - {file = "pluggy-0.13.1.tar.gz", hash = "sha256:15b2acde666561e1298d71b523007ed7364de07029219b604cf808bfa1c765b0"}, + {file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"}, + {file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"}, ] py = [ - {file = "py-1.10.0-py2.py3-none-any.whl", hash = "sha256:3b80836aa6d1feeaa108e046da6423ab8f6ceda6468545ae8d02d9d58d18818a"}, - {file = "py-1.10.0.tar.gz", hash = "sha256:21b81bda15b66ef5e1a777a21c4dcd9c20ad3efd0b3f817e7a809035269e1bd3"}, + {file = "py-1.11.0-py2.py3-none-any.whl", hash = "sha256:607c53218732647dff4acdfcd50cb62615cedf612e72d1724fb1a0cc6405b378"}, + {file = "py-1.11.0.tar.gz", hash = "sha256:51c75c4126074b472f746a24399ad32f6053d1b34b68d2fa41e558e6f4a98719"}, ] pycodestyle = [ - {file = "pycodestyle-2.7.0-py2.py3-none-any.whl", hash = "sha256:514f76d918fcc0b55c6680472f0a37970994e07bbb80725808c17089be302068"}, - {file = "pycodestyle-2.7.0.tar.gz", hash = "sha256:c389c1d06bf7904078ca03399a4816f974a1d590090fecea0c63ec26ebaf1cef"}, + {file = "pycodestyle-2.8.0-py2.py3-none-any.whl", hash = "sha256:720f8b39dde8b293825e7ff02c475f3077124006db4f440dcbc9a20b76548a20"}, + {file = "pycodestyle-2.8.0.tar.gz", hash = "sha256:eddd5847ef438ea1c7870ca7eb78a9d47ce0cdb4851a5523949f2601d0cbbe7f"}, ] pydantic = [ {file = "pydantic-1.8.2-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:05ddfd37c1720c392f4e0d43c484217b7521558302e7069ce8d318438d297739"}, @@ -1780,127 +1529,90 @@ pydocstyle = [ {file = "pydocstyle-6.1.1.tar.gz", hash = "sha256:1d41b7c459ba0ee6c345f2eb9ae827cab14a7533a88c5c6f7e94923f72df92dc"}, ] pyflakes = [ - {file = "pyflakes-2.3.1-py2.py3-none-any.whl", hash = "sha256:7893783d01b8a89811dd72d7dfd4d84ff098e5eed95cfa8905b22bbffe52efc3"}, - {file = "pyflakes-2.3.1.tar.gz", hash = "sha256:f5bc8ecabc05bb9d291eb5203d6810b49040f6ff446a756326104746cc00c1db"}, + {file = "pyflakes-2.4.0-py2.py3-none-any.whl", hash = "sha256:3bb3a3f256f4b7968c9c788781e4ff07dce46bdf12339dcda61053375426ee2e"}, + {file = "pyflakes-2.4.0.tar.gz", hash = "sha256:05a85c2872edf37a4ed30b0cce2f6093e1d0581f8c19d7393122da7e25b2b24c"}, ] pygments = [ - {file = "Pygments-2.10.0-py3-none-any.whl", hash = "sha256:b8e67fe6af78f492b3c4b3e2970c0624cbf08beb1e493b2c99b9fa1b67a20380"}, - {file = "Pygments-2.10.0.tar.gz", hash = "sha256:f398865f7eb6874156579fdf36bc840a03cab64d1cde9e93d68f46a425ec52c6"}, + {file = "Pygments-2.11.2-py3-none-any.whl", hash = "sha256:44238f1b60a76d78fc8ca0528ee429702aae011c265fe6a8dd8b63049ae41c65"}, + {file = "Pygments-2.11.2.tar.gz", hash = "sha256:4e426f72023d88d03b2fa258de560726ce890ff3b630f88c21cbb8b2503b8c6a"}, ] pymdown-extensions = [ - {file = "pymdown-extensions-8.2.tar.gz", hash = "sha256:b6daa94aad9e1310f9c64c8b1f01e4ce82937ab7eb53bfc92876a97aca02a6f4"}, - {file = "pymdown_extensions-8.2-py3-none-any.whl", hash = "sha256:141452d8ed61165518f2c923454bf054866b85cf466feedb0eb68f04acdc2560"}, + {file = "pymdown-extensions-9.3.tar.gz", hash = "sha256:a80553b243d3ed2d6c27723bcd64ca9887e560e6f4808baa96f36e93061eaf90"}, + {file = "pymdown_extensions-9.3-py3-none-any.whl", hash = "sha256:b37461a181c1c8103cfe1660081726a0361a8294cbfda88e5b02cefe976f0546"}, ] pyparsing = [ - {file = "pyparsing-2.4.7-py2.py3-none-any.whl", hash = "sha256:ef9d7589ef3c200abe66653d3f1ab1033c3c419ae9b9bdb1240a85b024efc88b"}, - {file = "pyparsing-2.4.7.tar.gz", hash = "sha256:c203ec8783bf771a155b207279b9bccb8dea02d8f0c9e5f8ead507bc3246ecc1"}, + {file = "pyparsing-3.0.7-py3-none-any.whl", hash = "sha256:a6c06a88f252e6c322f65faf8f418b16213b51bdfaece0524c1c1bc30c63c484"}, + {file = "pyparsing-3.0.7.tar.gz", hash = "sha256:18ee9022775d270c55187733956460083db60b37d0d0fb357445f3094eed3eea"}, ] pytest = [ {file = "pytest-6.2.5-py3-none-any.whl", hash = "sha256:7310f8d27bc79ced999e760ca304d69f6ba6c6649c0b60fb0e04a4a77cacc134"}, {file = "pytest-6.2.5.tar.gz", hash = "sha256:131b36680866a76e6781d13f101efb86cf674ebb9762eb70d3082b6f29889e89"}, ] pytest-asyncio = [ - {file = "pytest-asyncio-0.15.1.tar.gz", hash = "sha256:2564ceb9612bbd560d19ca4b41347b54e7835c2f792c504f698e05395ed63f6f"}, - {file = "pytest_asyncio-0.15.1-py3-none-any.whl", hash = "sha256:3042bcdf1c5d978f6b74d96a151c4cfb9dcece65006198389ccd7e6c60eb1eea"}, -] -pytest-clarity = [ - {file = "pytest-clarity-0.3.0a0.tar.gz", hash = "sha256:5cc99e3d9b7969dfe17e5f6072d45a917c59d363b679686d3c958a1ded2e4dcf"}, + {file = "pytest-asyncio-0.16.0.tar.gz", hash = "sha256:7496c5977ce88c34379df64a66459fe395cd05543f0a2f837016e7144391fcfb"}, + {file = "pytest_asyncio-0.16.0-py3-none-any.whl", hash = "sha256:5f2a21273c47b331ae6aa5b36087047b4899e40f03f18397c0e65fa5cca54e9b"}, ] pytest-cov = [ - {file = "pytest-cov-2.12.1.tar.gz", hash = "sha256:261ceeb8c227b726249b376b8526b600f38667ee314f910353fa318caa01f4d7"}, - {file = "pytest_cov-2.12.1-py2.py3-none-any.whl", hash = "sha256:261bb9e47e65bd099c89c3edf92972865210c36813f80ede5277dceb77a4a62a"}, + {file = "pytest-cov-3.0.0.tar.gz", hash = "sha256:e7f0f5b1617d2210a2cabc266dfe2f4c75a8d32fb89eafb7ad9d06f6d076d470"}, + {file = "pytest_cov-3.0.0-py3-none-any.whl", hash = "sha256:578d5d15ac4a25e5f961c938b85a05b09fdaae9deef3bb6de9a6e766622ca7a6"}, ] python-dateutil = [ {file = "python-dateutil-2.8.2.tar.gz", hash = "sha256:0123cacc1627ae19ddf3c27a5de5bd67ee4586fbdd6440d9748f8abb483d3e86"}, {file = "python_dateutil-2.8.2-py2.py3-none-any.whl", hash = "sha256:961d03dc3453ebbc59dbdea9e4e11c5651520a876d0f4db161e8674aae935da9"}, ] -python-multipart = [ - {file = "python-multipart-0.0.5.tar.gz", hash = "sha256:f7bb5f611fc600d15fa47b3974c8aa16e93724513b49b5f95c81e6624c83fa43"}, -] -pytkdocs = [ - {file = "pytkdocs-0.7.0-py3-none-any.whl", hash = "sha256:96c494143e70ccbb657bc4c0a93a97da0209f839f0236c08f227faedc51c1745"}, - {file = "pytkdocs-0.7.0.tar.gz", hash = "sha256:88c79290525f7658e8271ce19dd343c01c53bbe6c2801d1bfcc6792cad0636d5"}, +python-dotenv = [ + {file = "python-dotenv-0.19.2.tar.gz", hash = "sha256:a5de49a31e953b45ff2d2fd434bbc2670e8db5273606c1e737cc6b93eff3655f"}, + {file = "python_dotenv-0.19.2-py2.py3-none-any.whl", hash = "sha256:32b2bdc1873fd3a3c346da1c6db83d0053c3c62f28f1f38516070c4c8971b1d3"}, ] pyyaml = [ - {file = "PyYAML-5.4.1-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:3b2b1824fe7112845700f815ff6a489360226a5609b96ec2190a45e62a9fc922"}, - {file = "PyYAML-5.4.1-cp27-cp27m-win32.whl", hash = "sha256:129def1b7c1bf22faffd67b8f3724645203b79d8f4cc81f674654d9902cb4393"}, - {file = "PyYAML-5.4.1-cp27-cp27m-win_amd64.whl", hash = "sha256:4465124ef1b18d9ace298060f4eccc64b0850899ac4ac53294547536533800c8"}, - {file = "PyYAML-5.4.1-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:bb4191dfc9306777bc594117aee052446b3fa88737cd13b7188d0e7aa8162185"}, - {file = "PyYAML-5.4.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:6c78645d400265a062508ae399b60b8c167bf003db364ecb26dcab2bda048253"}, - {file = "PyYAML-5.4.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:4e0583d24c881e14342eaf4ec5fbc97f934b999a6828693a99157fde912540cc"}, - {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:72a01f726a9c7851ca9bfad6fd09ca4e090a023c00945ea05ba1638c09dc3347"}, - {file = "PyYAML-5.4.1-cp36-cp36m-manylinux2014_s390x.whl", hash = "sha256:895f61ef02e8fed38159bb70f7e100e00f471eae2bc838cd0f4ebb21e28f8541"}, - {file = "PyYAML-5.4.1-cp36-cp36m-win32.whl", hash = "sha256:3bd0e463264cf257d1ffd2e40223b197271046d09dadf73a0fe82b9c1fc385a5"}, - {file = "PyYAML-5.4.1-cp36-cp36m-win_amd64.whl", hash = "sha256:e4fac90784481d221a8e4b1162afa7c47ed953be40d31ab4629ae917510051df"}, - {file = "PyYAML-5.4.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:5accb17103e43963b80e6f837831f38d314a0495500067cb25afab2e8d7a4018"}, - {file = "PyYAML-5.4.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:e1d4970ea66be07ae37a3c2e48b5ec63f7ba6804bdddfdbd3cfd954d25a82e63"}, - {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:cb333c16912324fd5f769fff6bc5de372e9e7a202247b48870bc251ed40239aa"}, - {file = "PyYAML-5.4.1-cp37-cp37m-manylinux2014_s390x.whl", hash = "sha256:fe69978f3f768926cfa37b867e3843918e012cf83f680806599ddce33c2c68b0"}, - {file = "PyYAML-5.4.1-cp37-cp37m-win32.whl", hash = "sha256:dd5de0646207f053eb0d6c74ae45ba98c3395a571a2891858e87df7c9b9bd51b"}, - {file = "PyYAML-5.4.1-cp37-cp37m-win_amd64.whl", hash = "sha256:08682f6b72c722394747bddaf0aa62277e02557c0fd1c42cb853016a38f8dedf"}, - {file = "PyYAML-5.4.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:d2d9808ea7b4af864f35ea216be506ecec180628aced0704e34aca0b040ffe46"}, - {file = "PyYAML-5.4.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:8c1be557ee92a20f184922c7b6424e8ab6691788e6d86137c5d93c1a6ec1b8fb"}, - {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:fd7f6999a8070df521b6384004ef42833b9bd62cfee11a09bda1079b4b704247"}, - {file = "PyYAML-5.4.1-cp38-cp38-manylinux2014_s390x.whl", hash = "sha256:bfb51918d4ff3d77c1c856a9699f8492c612cde32fd3bcd344af9be34999bfdc"}, - {file = "PyYAML-5.4.1-cp38-cp38-win32.whl", hash = "sha256:fa5ae20527d8e831e8230cbffd9f8fe952815b2b7dae6ffec25318803a7528fc"}, - {file = "PyYAML-5.4.1-cp38-cp38-win_amd64.whl", hash = "sha256:0f5f5786c0e09baddcd8b4b45f20a7b5d61a7e7e99846e3c799b05c7c53fa696"}, - {file = "PyYAML-5.4.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:294db365efa064d00b8d1ef65d8ea2c3426ac366c0c4368d930bf1c5fb497f77"}, - {file = "PyYAML-5.4.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:74c1485f7707cf707a7aef42ef6322b8f97921bd89be2ab6317fd782c2d53183"}, - {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:d483ad4e639292c90170eb6f7783ad19490e7a8defb3e46f97dfe4bacae89122"}, - {file = "PyYAML-5.4.1-cp39-cp39-manylinux2014_s390x.whl", hash = "sha256:fdc842473cd33f45ff6bce46aea678a54e3d21f1b61a7750ce3c498eedfe25d6"}, - {file = "PyYAML-5.4.1-cp39-cp39-win32.whl", hash = "sha256:49d4cdd9065b9b6e206d0595fee27a96b5dd22618e7520c33204a4a3239d5b10"}, - {file = "PyYAML-5.4.1-cp39-cp39-win_amd64.whl", hash = "sha256:c20cfa2d49991c8b4147af39859b167664f2ad4561704ee74c1de03318e898db"}, - {file = "PyYAML-5.4.1.tar.gz", hash = "sha256:607774cbba28732bfa802b54baa7484215f530991055bb562efbed5b2f20a45e"}, + {file = "PyYAML-6.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:d4db7c7aef085872ef65a8fd7d6d09a14ae91f691dec3e87ee5ee0539d516f53"}, + {file = "PyYAML-6.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:9df7ed3b3d2e0ecfe09e14741b857df43adb5a3ddadc919a2d94fbdf78fea53c"}, + {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:77f396e6ef4c73fdc33a9157446466f1cff553d979bd00ecb64385760c6babdc"}, + {file = "PyYAML-6.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:a80a78046a72361de73f8f395f1f1e49f956c6be882eed58505a15f3e430962b"}, + {file = "PyYAML-6.0-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:f84fbc98b019fef2ee9a1cb3ce93e3187a6df0b2538a651bfb890254ba9f90b5"}, + {file = "PyYAML-6.0-cp310-cp310-win32.whl", hash = "sha256:2cd5df3de48857ed0544b34e2d40e9fac445930039f3cfe4bcc592a1f836d513"}, + {file = "PyYAML-6.0-cp310-cp310-win_amd64.whl", hash = "sha256:daf496c58a8c52083df09b80c860005194014c3698698d1a57cbcfa182142a3a"}, + {file = "PyYAML-6.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:897b80890765f037df3403d22bab41627ca8811ae55e9a722fd0392850ec4d86"}, + {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50602afada6d6cbfad699b0c7bb50d5ccffa7e46a3d738092afddc1f9758427f"}, + {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:48c346915c114f5fdb3ead70312bd042a953a8ce5c7106d5bfb1a5254e47da92"}, + {file = "PyYAML-6.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:98c4d36e99714e55cfbaaee6dd5badbc9a1ec339ebfc3b1f52e293aee6bb71a4"}, + {file = "PyYAML-6.0-cp36-cp36m-win32.whl", hash = "sha256:0283c35a6a9fbf047493e3a0ce8d79ef5030852c51e9d911a27badfde0605293"}, + {file = "PyYAML-6.0-cp36-cp36m-win_amd64.whl", hash = "sha256:07751360502caac1c067a8132d150cf3d61339af5691fe9e87803040dbc5db57"}, + {file = "PyYAML-6.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:819b3830a1543db06c4d4b865e70ded25be52a2e0631ccd2f6a47a2822f2fd7c"}, + {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:473f9edb243cb1935ab5a084eb238d842fb8f404ed2193a915d1784b5a6b5fc0"}, + {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:0ce82d761c532fe4ec3f87fc45688bdd3a4c1dc5e0b4a19814b9009a29baefd4"}, + {file = "PyYAML-6.0-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:231710d57adfd809ef5d34183b8ed1eeae3f76459c18fb4a0b373ad56bedcdd9"}, + {file = "PyYAML-6.0-cp37-cp37m-win32.whl", hash = "sha256:c5687b8d43cf58545ade1fe3e055f70eac7a5a1a0bf42824308d868289a95737"}, + {file = "PyYAML-6.0-cp37-cp37m-win_amd64.whl", hash = "sha256:d15a181d1ecd0d4270dc32edb46f7cb7733c7c508857278d3d378d14d606db2d"}, + {file = "PyYAML-6.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:0b4624f379dab24d3725ffde76559cff63d9ec94e1736b556dacdfebe5ab6d4b"}, + {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:213c60cd50106436cc818accf5baa1aba61c0189ff610f64f4a3e8c6726218ba"}, + {file = "PyYAML-6.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:9fa600030013c4de8165339db93d182b9431076eb98eb40ee068700c9c813e34"}, + {file = "PyYAML-6.0-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:277a0ef2981ca40581a47093e9e2d13b3f1fbbeffae064c1d21bfceba2030287"}, + {file = "PyYAML-6.0-cp38-cp38-win32.whl", hash = "sha256:d4eccecf9adf6fbcc6861a38015c2a64f38b9d94838ac1810a9023a0609e1b78"}, + {file = "PyYAML-6.0-cp38-cp38-win_amd64.whl", hash = "sha256:1e4747bc279b4f613a09eb64bba2ba602d8a6664c6ce6396a4d0cd413a50ce07"}, + {file = "PyYAML-6.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:055d937d65826939cb044fc8c9b08889e8c743fdc6a32b33e2390f66013e449b"}, + {file = "PyYAML-6.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e61ceaab6f49fb8bdfaa0f92c4b57bcfbea54c09277b1b4f7ac376bfb7a7c174"}, + {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d67d839ede4ed1b28a4e8909735fc992a923cdb84e618544973d7dfc71540803"}, + {file = "PyYAML-6.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:cba8c411ef271aa037d7357a2bc8f9ee8b58b9965831d9e51baf703280dc73d3"}, + {file = "PyYAML-6.0-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:40527857252b61eacd1d9af500c3337ba8deb8fc298940291486c465c8b46ec0"}, + {file = "PyYAML-6.0-cp39-cp39-win32.whl", hash = "sha256:b5b9eccad747aabaaffbc6064800670f0c297e52c12754eb1d976c57e4f74dcb"}, + {file = "PyYAML-6.0-cp39-cp39-win_amd64.whl", hash = "sha256:b3d267842bf12586ba6c734f89d1f5b871df0273157918b0ccefa29deb05c21c"}, + {file = "PyYAML-6.0.tar.gz", hash = "sha256:68fb519c14306fec9720a2a5b45bc9f0c8d1b9c72adf45c37baedfcd949c35a2"}, ] pyyaml-env-tag = [ {file = "pyyaml_env_tag-0.1-py3-none-any.whl", hash = "sha256:af31106dec8a4d68c60207c1886031cbf839b68aa7abccdb19868200532c2069"}, {file = "pyyaml_env_tag-0.1.tar.gz", hash = "sha256:70092675bda14fdec33b31ba77e7543de9ddc88f2e5b99160396572d11525bdb"}, ] -regex = [ - {file = "regex-2021.8.28-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:9d05ad5367c90814099000442b2125535e9d77581855b9bee8780f1b41f2b1a2"}, - {file = "regex-2021.8.28-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f3bf1bc02bc421047bfec3343729c4bbbea42605bcfd6d6bfe2c07ade8b12d2a"}, - {file = "regex-2021.8.28-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5f6a808044faae658f546dd5f525e921de9fa409de7a5570865467f03a626fc0"}, - {file = "regex-2021.8.28-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a617593aeacc7a691cc4af4a4410031654f2909053bd8c8e7db837f179a630eb"}, - {file = "regex-2021.8.28-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:79aef6b5cd41feff359acaf98e040844613ff5298d0d19c455b3d9ae0bc8c35a"}, - {file = "regex-2021.8.28-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:0fc1f8f06977c2d4f5e3d3f0d4a08089be783973fc6b6e278bde01f0544ff308"}, - {file = "regex-2021.8.28-cp310-cp310-win32.whl", hash = "sha256:6eebf512aa90751d5ef6a7c2ac9d60113f32e86e5687326a50d7686e309f66ed"}, - {file = "regex-2021.8.28-cp310-cp310-win_amd64.whl", hash = "sha256:ac88856a8cbccfc14f1b2d0b829af354cc1743cb375e7f04251ae73b2af6adf8"}, - {file = "regex-2021.8.28-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:c206587c83e795d417ed3adc8453a791f6d36b67c81416676cad053b4104152c"}, - {file = "regex-2021.8.28-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e8690ed94481f219a7a967c118abaf71ccc440f69acd583cab721b90eeedb77c"}, - {file = "regex-2021.8.28-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:328a1fad67445550b982caa2a2a850da5989fd6595e858f02d04636e7f8b0b13"}, - {file = "regex-2021.8.28-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c7cb4c512d2d3b0870e00fbbac2f291d4b4bf2634d59a31176a87afe2777c6f0"}, - {file = "regex-2021.8.28-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:66256b6391c057305e5ae9209941ef63c33a476b73772ca967d4a2df70520ec1"}, - {file = "regex-2021.8.28-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:8e44769068d33e0ea6ccdf4b84d80c5afffe5207aa4d1881a629cf0ef3ec398f"}, - {file = "regex-2021.8.28-cp36-cp36m-win32.whl", hash = "sha256:08d74bfaa4c7731b8dac0a992c63673a2782758f7cfad34cf9c1b9184f911354"}, - {file = "regex-2021.8.28-cp36-cp36m-win_amd64.whl", hash = "sha256:abb48494d88e8a82601af905143e0de838c776c1241d92021e9256d5515b3645"}, - {file = "regex-2021.8.28-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:b4c220a1fe0d2c622493b0a1fd48f8f991998fb447d3cd368033a4b86cf1127a"}, - {file = "regex-2021.8.28-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d4a332404baa6665b54e5d283b4262f41f2103c255897084ec8f5487ce7b9e8e"}, - {file = "regex-2021.8.28-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c61dcc1cf9fd165127a2853e2c31eb4fb961a4f26b394ac9fe5669c7a6592892"}, - {file = "regex-2021.8.28-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:ee329d0387b5b41a5dddbb6243a21cb7896587a651bebb957e2d2bb8b63c0791"}, - {file = "regex-2021.8.28-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:f60667673ff9c249709160529ab39667d1ae9fd38634e006bec95611f632e759"}, - {file = "regex-2021.8.28-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:b844fb09bd9936ed158ff9df0ab601e2045b316b17aa8b931857365ea8586906"}, - {file = "regex-2021.8.28-cp37-cp37m-win32.whl", hash = "sha256:4cde065ab33bcaab774d84096fae266d9301d1a2f5519d7bd58fc55274afbf7a"}, - {file = "regex-2021.8.28-cp37-cp37m-win_amd64.whl", hash = "sha256:1413b5022ed6ac0d504ba425ef02549a57d0f4276de58e3ab7e82437892704fc"}, - {file = "regex-2021.8.28-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:ed4b50355b066796dacdd1cf538f2ce57275d001838f9b132fab80b75e8c84dd"}, - {file = "regex-2021.8.28-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:28fc475f560d8f67cc8767b94db4c9440210f6958495aeae70fac8faec631797"}, - {file = "regex-2021.8.28-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bdc178caebd0f338d57ae445ef8e9b737ddf8fbc3ea187603f65aec5b041248f"}, - {file = "regex-2021.8.28-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:999ad08220467b6ad4bd3dd34e65329dd5d0df9b31e47106105e407954965256"}, - {file = "regex-2021.8.28-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:808ee5834e06f57978da3e003ad9d6292de69d2bf6263662a1a8ae30788e080b"}, - {file = "regex-2021.8.28-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d5111d4c843d80202e62b4fdbb4920db1dcee4f9366d6b03294f45ed7b18b42e"}, - {file = "regex-2021.8.28-cp38-cp38-win32.whl", hash = "sha256:473858730ef6d6ff7f7d5f19452184cd0caa062a20047f6d6f3e135a4648865d"}, - {file = "regex-2021.8.28-cp38-cp38-win_amd64.whl", hash = "sha256:31a99a4796bf5aefc8351e98507b09e1b09115574f7c9dbb9cf2111f7220d2e2"}, - {file = "regex-2021.8.28-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:04f6b9749e335bb0d2f68c707f23bb1773c3fb6ecd10edf0f04df12a8920d468"}, - {file = "regex-2021.8.28-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9b006628fe43aa69259ec04ca258d88ed19b64791693df59c422b607b6ece8bb"}, - {file = "regex-2021.8.28-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:121f4b3185feaade3f85f70294aef3f777199e9b5c0c0245c774ae884b110a2d"}, - {file = "regex-2021.8.28-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:a577a21de2ef8059b58f79ff76a4da81c45a75fe0bfb09bc8b7bb4293fa18983"}, - {file = "regex-2021.8.28-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1743345e30917e8c574f273f51679c294effba6ad372db1967852f12c76759d8"}, - {file = "regex-2021.8.28-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:e1e8406b895aba6caa63d9fd1b6b1700d7e4825f78ccb1e5260551d168db38ed"}, - {file = "regex-2021.8.28-cp39-cp39-win32.whl", hash = "sha256:ed283ab3a01d8b53de3a05bfdf4473ae24e43caee7dcb5584e86f3f3e5ab4374"}, - {file = "regex-2021.8.28-cp39-cp39-win_amd64.whl", hash = "sha256:610b690b406653c84b7cb6091facb3033500ee81089867ee7d59e675f9ca2b73"}, - {file = "regex-2021.8.28.tar.gz", hash = "sha256:f585cbbeecb35f35609edccb95efd95a3e35824cd7752b586503f7e6087303f1"}, +requests = [ + {file = "requests-2.26.0-py2.py3-none-any.whl", hash = "sha256:6c1246513ecd5ecd4528a0906f910e8f0f9c6b8ec72030dc9fd154dc1a6efd24"}, + {file = "requests-2.26.0.tar.gz", hash = "sha256:b8aa58f8cf793ffd8782d3d8cb19e66ef36f7aba4353eec859e74678b01b07a7"}, +] +respx = [ + {file = "respx-0.19.0-py2.py3-none-any.whl", hash = "sha256:1ac1cc99bf892ffd3e33108ae43d71d8309a58ac226965f4bd81ec055600f265"}, + {file = "respx-0.19.0.tar.gz", hash = "sha256:4a09e15803c7450d45303520ec528794c9fd77b05984263bc83b78aabbb39413"}, ] restructuredtext-lint = [ - {file = "restructuredtext_lint-1.3.2.tar.gz", hash = "sha256:d3b10a1fe2ecac537e51ae6d151b223b78de9fafdd50e5eb6b08c243df173c80"}, + {file = "restructuredtext_lint-1.4.0.tar.gz", hash = "sha256:1b235c0c922341ab6c530390892eb9e92f90b9b75046063e047cacfb0f050c45"}, ] rfc3986 = [ {file = "rfc3986-1.5.0-py2.py3-none-any.whl", hash = "sha256:a86d6e1f5b1dc238b218b012df0aa79409667bb209e58da56d0b94704e712a97"}, @@ -1911,163 +1623,87 @@ six = [ {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"}, ] smmap = [ - {file = "smmap-4.0.0-py2.py3-none-any.whl", hash = "sha256:a9a7479e4c572e2e775c404dcd3080c8dc49f39918c2cf74913d30c4c478e3c2"}, - {file = "smmap-4.0.0.tar.gz", hash = "sha256:7e65386bd122d45405ddf795637b7f7d2b532e7e401d46bbe3fb49b9986d5182"}, + {file = "smmap-5.0.0-py3-none-any.whl", hash = "sha256:2aba19d6a040e78d8b09de5c57e96207b09ed71d8e55ce0959eeee6c8e190d94"}, + {file = "smmap-5.0.0.tar.gz", hash = "sha256:c840e62059cd3be204b0c9c9f74be2c09d5648eddd4580d9314c3ecde0b30936"}, ] sniffio = [ {file = "sniffio-1.2.0-py3-none-any.whl", hash = "sha256:471b71698eac1c2112a40ce2752bb2f4a4814c22a54a3eed3676bc0f5ca9f663"}, {file = "sniffio-1.2.0.tar.gz", hash = "sha256:c4666eecec1d3f50960c6bdf61ab7bc350648da6c126e3cf6898d8cd4ddcd3de"}, ] snowballstemmer = [ - {file = "snowballstemmer-2.1.0-py2.py3-none-any.whl", hash = "sha256:b51b447bea85f9968c13b650126a888aabd4cb4463fca868ec596826325dedc2"}, - {file = "snowballstemmer-2.1.0.tar.gz", hash = "sha256:e997baa4f2e9139951b6f4c631bad912dfd3c792467e2f03d7239464af90e914"}, + {file = "snowballstemmer-2.2.0-py2.py3-none-any.whl", hash = "sha256:c8e1716e83cc398ae16824e5572ae04e0d9fc2c6b985fb0f900f5f0c96ecba1a"}, + {file = "snowballstemmer-2.2.0.tar.gz", hash = "sha256:09b16deb8547d3412ad7b590689584cd0fe25ec8db3be37788be3810cbf19cb1"}, ] starlette = [ - {file = "starlette-0.16.0-py3-none-any.whl", hash = "sha256:38eb24bf705a2c317e15868e384c1b8a12ca396e5a3c3a003db7e667c43f939f"}, - {file = "starlette-0.16.0.tar.gz", hash = "sha256:e1904b5d0007aee24bdd3c43994be9b3b729f4f58e740200de1d623f8c3a8870"}, + {file = "starlette-0.17.1-py3-none-any.whl", hash = "sha256:26a18cbda5e6b651c964c12c88b36d9898481cd428ed6e063f5f29c418f73050"}, + {file = "starlette-0.17.1.tar.gz", hash = "sha256:57eab3cc975a28af62f6faec94d355a410634940f10b30d68d31cb5ec1b44ae8"}, ] stevedore = [ - {file = "stevedore-3.4.0-py3-none-any.whl", hash = "sha256:920ce6259f0b2498aaa4545989536a27e4e4607b8318802d7ddc3a533d3d069e"}, - {file = "stevedore-3.4.0.tar.gz", hash = "sha256:59b58edb7f57b11897f150475e7bc0c39c5381f0b8e3fa9f5c20ce6c89ec4aa1"}, -] -termcolor = [ - {file = "termcolor-1.1.0.tar.gz", hash = "sha256:1d6d69ce66211143803fbc56652b41d73b4a400a2891d7bf7a1cdf4c02de613b"}, + {file = "stevedore-3.5.0-py3-none-any.whl", hash = "sha256:a547de73308fd7e90075bb4d301405bebf705292fa90a90fc3bcf9133f58616c"}, + {file = "stevedore-3.5.0.tar.gz", hash = "sha256:f40253887d8712eaa2bb0ea3830374416736dc8ec0e22f5a65092c1174c44335"}, ] testfixtures = [ - {file = "testfixtures-6.18.1-py2.py3-none-any.whl", hash = "sha256:486be7b01eb71326029811878a3317b7e7994324621c0ec633c8e24499d8d5b3"}, - {file = "testfixtures-6.18.1.tar.gz", hash = "sha256:0a6422737f6d89b45cdef1e2df5576f52ad0f507956002ce1020daa9f44211d6"}, + {file = "testfixtures-6.18.5-py2.py3-none-any.whl", hash = "sha256:7de200e24f50a4a5d6da7019fb1197aaf5abd475efb2ec2422fdcf2f2eb98c1d"}, + {file = "testfixtures-6.18.5.tar.gz", hash = "sha256:02dae883f567f5b70fd3ad3c9eefb95912e78ac90be6c7444b5e2f46bf572c84"}, ] tokenize-rt = [ - {file = "tokenize_rt-4.1.0-py2.py3-none-any.whl", hash = "sha256:b37251fa28c21e8cce2e42f7769a35fba2dd2ecafb297208f9a9a8add3ca7793"}, - {file = "tokenize_rt-4.1.0.tar.gz", hash = "sha256:ab339b5ff829eb5e198590477f9c03c84e762b3e455e74c018956e7e326cbc70"}, + {file = "tokenize_rt-4.2.1-py2.py3-none-any.whl", hash = "sha256:08a27fa032a81cf45e8858d0ac706004fcd523e8463415ddf1442be38e204ea8"}, + {file = "tokenize_rt-4.2.1.tar.gz", hash = "sha256:0d4f69026fed520f8a1e0103aa36c406ef4661417f20ca643f913e33531b3b94"}, ] toml = [ {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"}, {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"}, ] -tornado = [ - {file = "tornado-6.1-cp35-cp35m-macosx_10_9_x86_64.whl", hash = "sha256:d371e811d6b156d82aa5f9a4e08b58debf97c302a35714f6f45e35139c332e32"}, - {file = "tornado-6.1-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:0d321a39c36e5f2c4ff12b4ed58d41390460f798422c4504e09eb5678e09998c"}, - {file = "tornado-6.1-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:9de9e5188a782be6b1ce866e8a51bc76a0fbaa0e16613823fc38e4fc2556ad05"}, - {file = "tornado-6.1-cp35-cp35m-manylinux2010_i686.whl", hash = "sha256:61b32d06ae8a036a6607805e6720ef00a3c98207038444ba7fd3d169cd998910"}, - {file = "tornado-6.1-cp35-cp35m-manylinux2010_x86_64.whl", hash = "sha256:3e63498f680547ed24d2c71e6497f24bca791aca2fe116dbc2bd0ac7f191691b"}, - {file = "tornado-6.1-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:6c77c9937962577a6a76917845d06af6ab9197702a42e1346d8ae2e76b5e3675"}, - {file = "tornado-6.1-cp35-cp35m-win32.whl", hash = "sha256:6286efab1ed6e74b7028327365cf7346b1d777d63ab30e21a0f4d5b275fc17d5"}, - {file = "tornado-6.1-cp35-cp35m-win_amd64.whl", hash = "sha256:fa2ba70284fa42c2a5ecb35e322e68823288a4251f9ba9cc77be04ae15eada68"}, - {file = "tornado-6.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:0a00ff4561e2929a2c37ce706cb8233b7907e0cdc22eab98888aca5dd3775feb"}, - {file = "tornado-6.1-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:748290bf9112b581c525e6e6d3820621ff020ed95af6f17fedef416b27ed564c"}, - {file = "tornado-6.1-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:e385b637ac3acaae8022e7e47dfa7b83d3620e432e3ecb9a3f7f58f150e50921"}, - {file = "tornado-6.1-cp36-cp36m-manylinux2010_i686.whl", hash = "sha256:25ad220258349a12ae87ede08a7b04aca51237721f63b1808d39bdb4b2164558"}, - {file = "tornado-6.1-cp36-cp36m-manylinux2010_x86_64.whl", hash = "sha256:65d98939f1a2e74b58839f8c4dab3b6b3c1ce84972ae712be02845e65391ac7c"}, - {file = "tornado-6.1-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:e519d64089b0876c7b467274468709dadf11e41d65f63bba207e04217f47c085"}, - {file = "tornado-6.1-cp36-cp36m-win32.whl", hash = "sha256:b87936fd2c317b6ee08a5741ea06b9d11a6074ef4cc42e031bc6403f82a32575"}, - {file = "tornado-6.1-cp36-cp36m-win_amd64.whl", hash = "sha256:cc0ee35043162abbf717b7df924597ade8e5395e7b66d18270116f8745ceb795"}, - {file = "tornado-6.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:7250a3fa399f08ec9cb3f7b1b987955d17e044f1ade821b32e5f435130250d7f"}, - {file = "tornado-6.1-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:ed3ad863b1b40cd1d4bd21e7498329ccaece75db5a5bf58cd3c9f130843e7102"}, - {file = "tornado-6.1-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:dcef026f608f678c118779cd6591c8af6e9b4155c44e0d1bc0c87c036fb8c8c4"}, - {file = "tornado-6.1-cp37-cp37m-manylinux2010_i686.whl", hash = "sha256:70dec29e8ac485dbf57481baee40781c63e381bebea080991893cd297742b8fd"}, - {file = "tornado-6.1-cp37-cp37m-manylinux2010_x86_64.whl", hash = "sha256:d3f7594930c423fd9f5d1a76bee85a2c36fd8b4b16921cae7e965f22575e9c01"}, - {file = "tornado-6.1-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:3447475585bae2e77ecb832fc0300c3695516a47d46cefa0528181a34c5b9d3d"}, - {file = "tornado-6.1-cp37-cp37m-win32.whl", hash = "sha256:e7229e60ac41a1202444497ddde70a48d33909e484f96eb0da9baf8dc68541df"}, - {file = "tornado-6.1-cp37-cp37m-win_amd64.whl", hash = "sha256:cb5ec8eead331e3bb4ce8066cf06d2dfef1bfb1b2a73082dfe8a161301b76e37"}, - {file = "tornado-6.1-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:20241b3cb4f425e971cb0a8e4ffc9b0a861530ae3c52f2b0434e6c1b57e9fd95"}, - {file = "tornado-6.1-cp38-cp38-manylinux1_i686.whl", hash = "sha256:c77da1263aa361938476f04c4b6c8916001b90b2c2fdd92d8d535e1af48fba5a"}, - {file = "tornado-6.1-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:fba85b6cd9c39be262fcd23865652920832b61583de2a2ca907dbd8e8a8c81e5"}, - {file = "tornado-6.1-cp38-cp38-manylinux2010_i686.whl", hash = "sha256:1e8225a1070cd8eec59a996c43229fe8f95689cb16e552d130b9793cb570a288"}, - {file = "tornado-6.1-cp38-cp38-manylinux2010_x86_64.whl", hash = "sha256:d14d30e7f46a0476efb0deb5b61343b1526f73ebb5ed84f23dc794bdb88f9d9f"}, - {file = "tornado-6.1-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:8f959b26f2634a091bb42241c3ed8d3cedb506e7c27b8dd5c7b9f745318ddbb6"}, - {file = "tornado-6.1-cp38-cp38-win32.whl", hash = "sha256:34ca2dac9e4d7afb0bed4677512e36a52f09caa6fded70b4e3e1c89dbd92c326"}, - {file = "tornado-6.1-cp38-cp38-win_amd64.whl", hash = "sha256:6196a5c39286cc37c024cd78834fb9345e464525d8991c21e908cc046d1cc02c"}, - {file = "tornado-6.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f0ba29bafd8e7e22920567ce0d232c26d4d47c8b5cf4ed7b562b5db39fa199c5"}, - {file = "tornado-6.1-cp39-cp39-manylinux1_i686.whl", hash = "sha256:33892118b165401f291070100d6d09359ca74addda679b60390b09f8ef325ffe"}, - {file = "tornado-6.1-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:7da13da6f985aab7f6f28debab00c67ff9cbacd588e8477034c0652ac141feea"}, - {file = "tornado-6.1-cp39-cp39-manylinux2010_i686.whl", hash = "sha256:e0791ac58d91ac58f694d8d2957884df8e4e2f6687cdf367ef7eb7497f79eaa2"}, - {file = "tornado-6.1-cp39-cp39-manylinux2010_x86_64.whl", hash = "sha256:66324e4e1beede9ac79e60f88de548da58b1f8ab4b2f1354d8375774f997e6c0"}, - {file = "tornado-6.1-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:a48900ecea1cbb71b8c71c620dee15b62f85f7c14189bdeee54966fbd9a0c5bd"}, - {file = "tornado-6.1-cp39-cp39-win32.whl", hash = "sha256:d3d20ea5782ba63ed13bc2b8c291a053c8d807a8fa927d941bd718468f7b950c"}, - {file = "tornado-6.1-cp39-cp39-win_amd64.whl", hash = "sha256:548430be2740e327b3fe0201abe471f314741efcb0067ec4f2d7dcfb4825f3e4"}, - {file = "tornado-6.1.tar.gz", hash = "sha256:33c6e81d7bd55b468d2e793517c909b139960b6c790a60b7991b9b6b76fb9791"}, -] -typed-ast = [ - {file = "typed_ast-1.4.3-cp35-cp35m-manylinux1_i686.whl", hash = "sha256:2068531575a125b87a41802130fa7e29f26c09a2833fea68d9a40cf33902eba6"}, - {file = "typed_ast-1.4.3-cp35-cp35m-manylinux1_x86_64.whl", hash = "sha256:c907f561b1e83e93fad565bac5ba9c22d96a54e7ea0267c708bffe863cbe4075"}, - {file = "typed_ast-1.4.3-cp35-cp35m-manylinux2014_aarch64.whl", hash = "sha256:1b3ead4a96c9101bef08f9f7d1217c096f31667617b58de957f690c92378b528"}, - {file = "typed_ast-1.4.3-cp35-cp35m-win32.whl", hash = "sha256:dde816ca9dac1d9c01dd504ea5967821606f02e510438120091b84e852367428"}, - {file = "typed_ast-1.4.3-cp35-cp35m-win_amd64.whl", hash = "sha256:777a26c84bea6cd934422ac2e3b78863a37017618b6e5c08f92ef69853e765d3"}, - {file = "typed_ast-1.4.3-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f8afcf15cc511ada719a88e013cec87c11aff7b91f019295eb4530f96fe5ef2f"}, - {file = "typed_ast-1.4.3-cp36-cp36m-manylinux1_i686.whl", hash = "sha256:52b1eb8c83f178ab787f3a4283f68258525f8d70f778a2f6dd54d3b5e5fb4341"}, - {file = "typed_ast-1.4.3-cp36-cp36m-manylinux1_x86_64.whl", hash = "sha256:01ae5f73431d21eead5015997ab41afa53aa1fbe252f9da060be5dad2c730ace"}, - {file = "typed_ast-1.4.3-cp36-cp36m-manylinux2014_aarch64.whl", hash = "sha256:c190f0899e9f9f8b6b7863debfb739abcb21a5c054f911ca3596d12b8a4c4c7f"}, - {file = "typed_ast-1.4.3-cp36-cp36m-win32.whl", hash = "sha256:398e44cd480f4d2b7ee8d98385ca104e35c81525dd98c519acff1b79bdaac363"}, - {file = "typed_ast-1.4.3-cp36-cp36m-win_amd64.whl", hash = "sha256:bff6ad71c81b3bba8fa35f0f1921fb24ff4476235a6e94a26ada2e54370e6da7"}, - {file = "typed_ast-1.4.3-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0fb71b8c643187d7492c1f8352f2c15b4c4af3f6338f21681d3681b3dc31a266"}, - {file = "typed_ast-1.4.3-cp37-cp37m-manylinux1_i686.whl", hash = "sha256:760ad187b1041a154f0e4d0f6aae3e40fdb51d6de16e5c99aedadd9246450e9e"}, - {file = "typed_ast-1.4.3-cp37-cp37m-manylinux1_x86_64.whl", hash = "sha256:5feca99c17af94057417d744607b82dd0a664fd5e4ca98061480fd8b14b18d04"}, - {file = "typed_ast-1.4.3-cp37-cp37m-manylinux2014_aarch64.whl", hash = "sha256:95431a26309a21874005845c21118c83991c63ea800dd44843e42a916aec5899"}, - {file = "typed_ast-1.4.3-cp37-cp37m-win32.whl", hash = "sha256:aee0c1256be6c07bd3e1263ff920c325b59849dc95392a05f258bb9b259cf39c"}, - {file = "typed_ast-1.4.3-cp37-cp37m-win_amd64.whl", hash = "sha256:9ad2c92ec681e02baf81fdfa056fe0d818645efa9af1f1cd5fd6f1bd2bdfd805"}, - {file = "typed_ast-1.4.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:b36b4f3920103a25e1d5d024d155c504080959582b928e91cb608a65c3a49e1a"}, - {file = "typed_ast-1.4.3-cp38-cp38-manylinux1_i686.whl", hash = "sha256:067a74454df670dcaa4e59349a2e5c81e567d8d65458d480a5b3dfecec08c5ff"}, - {file = "typed_ast-1.4.3-cp38-cp38-manylinux1_x86_64.whl", hash = "sha256:7538e495704e2ccda9b234b82423a4038f324f3a10c43bc088a1636180f11a41"}, - {file = "typed_ast-1.4.3-cp38-cp38-manylinux2014_aarch64.whl", hash = "sha256:af3d4a73793725138d6b334d9d247ce7e5f084d96284ed23f22ee626a7b88e39"}, - {file = "typed_ast-1.4.3-cp38-cp38-win32.whl", hash = "sha256:f2362f3cb0f3172c42938946dbc5b7843c2a28aec307c49100c8b38764eb6927"}, - {file = "typed_ast-1.4.3-cp38-cp38-win_amd64.whl", hash = "sha256:dd4a21253f42b8d2b48410cb31fe501d32f8b9fbeb1f55063ad102fe9c425e40"}, - {file = "typed_ast-1.4.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f328adcfebed9f11301eaedfa48e15bdece9b519fb27e6a8c01aa52a17ec31b3"}, - {file = "typed_ast-1.4.3-cp39-cp39-manylinux1_i686.whl", hash = "sha256:2c726c276d09fc5c414693a2de063f521052d9ea7c240ce553316f70656c84d4"}, - {file = "typed_ast-1.4.3-cp39-cp39-manylinux1_x86_64.whl", hash = "sha256:cae53c389825d3b46fb37538441f75d6aecc4174f615d048321b716df2757fb0"}, - {file = "typed_ast-1.4.3-cp39-cp39-manylinux2014_aarch64.whl", hash = "sha256:b9574c6f03f685070d859e75c7f9eeca02d6933273b5e69572e5ff9d5e3931c3"}, - {file = "typed_ast-1.4.3-cp39-cp39-win32.whl", hash = "sha256:209596a4ec71d990d71d5e0d312ac935d86930e6eecff6ccc7007fe54d703808"}, - {file = "typed_ast-1.4.3-cp39-cp39-win_amd64.whl", hash = "sha256:9c6d1a54552b5330bc657b7ef0eae25d00ba7ffe85d9ea8ae6540d2197a3788c"}, - {file = "typed_ast-1.4.3.tar.gz", hash = "sha256:fb1bbeac803adea29cedd70781399c99138358c26d05fcbd23c13016b7f5ec65"}, +tomli = [ + {file = "tomli-1.2.3-py3-none-any.whl", hash = "sha256:e3069e4be3ead9668e21cb9b074cd948f7b3113fd9c8bba083f48247aab8b11c"}, + {file = "tomli-1.2.3.tar.gz", hash = "sha256:05b6166bff487dc068d322585c7ea4ef78deed501cc124060e0f238e89a9231f"}, ] typing-extensions = [ - {file = "typing_extensions-3.10.0.2-py2-none-any.whl", hash = "sha256:d8226d10bc02a29bcc81df19a26e56a9647f8b0a6d4a83924139f4a8b01f17b7"}, - {file = "typing_extensions-3.10.0.2-py3-none-any.whl", hash = "sha256:f1d25edafde516b146ecd0613dabcc61409817af4766fbbcfb8d1ad4ec441a34"}, - {file = "typing_extensions-3.10.0.2.tar.gz", hash = "sha256:49f75d16ff11f1cd258e1b988ccff82a3ca5570217d7ad8c5f48205dd99a677e"}, + {file = "typing_extensions-4.1.1-py3-none-any.whl", hash = "sha256:21c85e0fe4b9a155d0799430b0ad741cdce7e359660ccbd8b530613e8df88ce2"}, + {file = "typing_extensions-4.1.1.tar.gz", hash = "sha256:1a9462dcc3347a79b1f1c0271fbe79e844580bb598bafa1ed208b94da3cdcd42"}, ] -typing-inspect = [ - {file = "typing_inspect-0.6.0-py2-none-any.whl", hash = "sha256:de08f50a22955ddec353876df7b2545994d6df08a2f45d54ac8c05e530372ca0"}, - {file = "typing_inspect-0.6.0-py3-none-any.whl", hash = "sha256:3b98390df4d999a28cf5b35d8b333425af5da2ece8a4ea9e98f71e7591347b4f"}, - {file = "typing_inspect-0.6.0.tar.gz", hash = "sha256:8f1b1dd25908dbfd81d3bebc218011531e7ab614ba6e5bf7826d887c834afab7"}, +urllib3 = [ + {file = "urllib3-1.26.9-py2.py3-none-any.whl", hash = "sha256:44ece4d53fb1706f667c9bd1c648f5469a2ec925fcf3a776667042d645472c14"}, + {file = "urllib3-1.26.9.tar.gz", hash = "sha256:aabaf16477806a5e1dd19aa41f8c2b7950dd3c746362d7e3223dbe6de6ac448e"}, ] -virtualenv = [ - {file = "virtualenv-20.7.2-py2.py3-none-any.whl", hash = "sha256:e4670891b3a03eb071748c569a87cceaefbf643c5bac46d996c5a45c34aa0f06"}, - {file = "virtualenv-20.7.2.tar.gz", hash = "sha256:9ef4e8ee4710826e98ff3075c9a4739e2cb1040de6a2a8d35db0055840dc96a0"}, +uvicorn = [ + {file = "uvicorn-0.16.0-py3-none-any.whl", hash = "sha256:d8c839231f270adaa6d338d525e2652a0b4a5f4c2430b5c4ef6ae4d11776b0d2"}, + {file = "uvicorn-0.16.0.tar.gz", hash = "sha256:eacb66afa65e0648fcbce5e746b135d09722231ffffc61883d4fac2b62fbea8d"}, ] watchdog = [ - {file = "watchdog-2.1.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:5f57ce4f7e498278fb2a091f39359930144a0f2f90ea8cbf4523c4e25de34028"}, - {file = "watchdog-2.1.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8b74d0d92a69a7ab5f101f9fe74e44ba017be269efa824337366ccbb4effde85"}, - {file = "watchdog-2.1.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:59767f476cd1f48531bf378f0300565d879688c82da8369ca8c52f633299523c"}, - {file = "watchdog-2.1.5-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:814d396859c95598f7576d15bc257c3bd3ba61fa4bc1db7dfc18f09070ded7da"}, - {file = "watchdog-2.1.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:28777dbed3bbd95f9c70f461443990a36c07dbf49ae7cd69932cdd1b8fb2850c"}, - {file = "watchdog-2.1.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5cf78f794c9d7bc64a626ef4f71aff88f57a7ae288e0b359a9c6ea711a41395f"}, - {file = "watchdog-2.1.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:43bf728eb7830559f329864ab5da2302c15b2efbac24ad84ccc09949ba753c40"}, - {file = "watchdog-2.1.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:a7053d4d22dc95c5e0c90aeeae1e4ed5269d2f04001798eec43a654a03008d22"}, - {file = "watchdog-2.1.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:6f3ad1d973fe8fc8fe64ba38f6a934b74346342fa98ef08ad5da361a05d46044"}, - {file = "watchdog-2.1.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:41d44ef21a77a32b55ce9bf59b75777063751f688de51098859b7c7f6466589a"}, - {file = "watchdog-2.1.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ed4ca4351cd2bb0d863ee737a2011ca44d8d8be19b43509bd4507f8a449b376b"}, - {file = "watchdog-2.1.5-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:8874d5ad6b7f43b18935d9b0183e29727a623a216693d6938d07dfd411ba462f"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_aarch64.whl", hash = "sha256:50a7f81f99d238f72185f481b493f9de80096e046935b60ea78e1276f3d76960"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_armv7l.whl", hash = "sha256:e40e33a4889382824846b4baa05634e1365b47c6fa40071dc2d06b4d7c715fc1"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_i686.whl", hash = "sha256:78b1514067ff4089f4dac930b043a142997a5b98553120919005e97fbaba6546"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_ppc64.whl", hash = "sha256:58ae842300cbfe5e62fb068c83901abe76e4f413234b7bec5446e4275eb1f9cb"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:b0cc7d8b7d60da6c313779d85903ce39a63d89d866014b085f720a083d5f3e9a"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_s390x.whl", hash = "sha256:e60d3bb7166b7cb830b86938d1eb0e6cfe23dfd634cce05c128f8f9967895193"}, - {file = "watchdog-2.1.5-py3-none-manylinux2014_x86_64.whl", hash = "sha256:51af09ae937ada0e9a10cc16988ec03c649754a91526170b6839b89fc56d6acb"}, - {file = "watchdog-2.1.5-py3-none-win32.whl", hash = "sha256:9391003635aa783957b9b11175d9802d3272ed67e69ef2e3394c0b6d9d24fa9a"}, - {file = "watchdog-2.1.5-py3-none-win_amd64.whl", hash = "sha256:eab14adfc417c2c983fbcb2c73ef3f28ba6990d1fff45d1180bf7e38bda0d98d"}, - {file = "watchdog-2.1.5-py3-none-win_ia64.whl", hash = "sha256:a2888a788893c4ef7e562861ec5433875b7915f930a5a7ed3d32c048158f1be5"}, - {file = "watchdog-2.1.5.tar.gz", hash = "sha256:5563b005907613430ef3d4aaac9c78600dd5704e84764cb6deda4b3d72807f09"}, + {file = "watchdog-2.1.6-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:9693f35162dc6208d10b10ddf0458cc09ad70c30ba689d9206e02cd836ce28a3"}, + {file = "watchdog-2.1.6-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:aba5c812f8ee8a3ff3be51887ca2d55fb8e268439ed44110d3846e4229eb0e8b"}, + {file = "watchdog-2.1.6-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:4ae38bf8ba6f39d5b83f78661273216e7db5b00f08be7592062cb1fc8b8ba542"}, + {file = "watchdog-2.1.6-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:ad6f1796e37db2223d2a3f302f586f74c72c630b48a9872c1e7ae8e92e0ab669"}, + {file = "watchdog-2.1.6-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:922a69fa533cb0c793b483becaaa0845f655151e7256ec73630a1b2e9ebcb660"}, + {file = "watchdog-2.1.6-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:b2fcf9402fde2672545b139694284dc3b665fd1be660d73eca6805197ef776a3"}, + {file = "watchdog-2.1.6-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:3386b367e950a11b0568062b70cc026c6f645428a698d33d39e013aaeda4cc04"}, + {file = "watchdog-2.1.6-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:8f1c00aa35f504197561060ca4c21d3cc079ba29cf6dd2fe61024c70160c990b"}, + {file = "watchdog-2.1.6-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:b52b88021b9541a60531142b0a451baca08d28b74a723d0c99b13c8c8d48d604"}, + {file = "watchdog-2.1.6-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:8047da932432aa32c515ec1447ea79ce578d0559362ca3605f8e9568f844e3c6"}, + {file = "watchdog-2.1.6-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:e92c2d33858c8f560671b448205a268096e17870dcf60a9bb3ac7bfbafb7f5f9"}, + {file = "watchdog-2.1.6-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:b7d336912853d7b77f9b2c24eeed6a5065d0a0cc0d3b6a5a45ad6d1d05fb8cd8"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_aarch64.whl", hash = "sha256:cca7741c0fcc765568350cb139e92b7f9f3c9a08c4f32591d18ab0a6ac9e71b6"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_armv7l.whl", hash = "sha256:25fb5240b195d17de949588628fdf93032ebf163524ef08933db0ea1f99bd685"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_i686.whl", hash = "sha256:be9be735f827820a06340dff2ddea1fb7234561fa5e6300a62fe7f54d40546a0"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_ppc64.whl", hash = "sha256:d0d19fb2441947b58fbf91336638c2b9f4cc98e05e1045404d7a4cb7cddc7a65"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_ppc64le.whl", hash = "sha256:3becdb380d8916c873ad512f1701f8a92ce79ec6978ffde92919fd18d41da7fb"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_s390x.whl", hash = "sha256:ae67501c95606072aafa865b6ed47343ac6484472a2f95490ba151f6347acfc2"}, + {file = "watchdog-2.1.6-py3-none-manylinux2014_x86_64.whl", hash = "sha256:e0f30db709c939cabf64a6dc5babb276e6d823fd84464ab916f9b9ba5623ca15"}, + {file = "watchdog-2.1.6-py3-none-win32.whl", hash = "sha256:e02794ac791662a5eafc6ffeaf9bcc149035a0e48eb0a9d40a8feb4622605a3d"}, + {file = "watchdog-2.1.6-py3-none-win_amd64.whl", hash = "sha256:bd9ba4f332cf57b2c1f698be0728c020399ef3040577cde2939f2e045b39c1e5"}, + {file = "watchdog-2.1.6-py3-none-win_ia64.whl", hash = "sha256:a0f1c7edf116a12f7245be06120b1852275f9506a7d90227648b250755a03923"}, + {file = "watchdog-2.1.6.tar.gz", hash = "sha256:a36e75df6c767cbf46f61a91c70b3ba71811dfa0aca4a324d9407a06a8b7a2e7"}, ] wemake-python-styleguide = [ - {file = "wemake-python-styleguide-0.15.3.tar.gz", hash = "sha256:8b89aedabae67b7b915908ed06c178b702068137c0d8afe1fb59cdc829cd2143"}, - {file = "wemake_python_styleguide-0.15.3-py3-none-any.whl", hash = "sha256:a382f6c9ec87d56daa08a11e47cab019c99b384f1393b32564ebc74c6da80441"}, + {file = "wemake-python-styleguide-0.16.0.tar.gz", hash = "sha256:3bf0a4962404e6fd6fa479e72e2ba3fb75d5920ea6c44b72b45240c9e519543c"}, + {file = "wemake_python_styleguide-0.16.0-py3-none-any.whl", hash = "sha256:8caa92b4aa77b08a505d718553238812d1b612b1036bc171ca3aa18345efe0b4"}, ] win32-setctime = [ - {file = "win32_setctime-1.0.3-py3-none-any.whl", hash = "sha256:dc925662de0a6eb987f0b01f599c01a8236cb8c62831c22d9cada09ad958243e"}, - {file = "win32_setctime-1.0.3.tar.gz", hash = "sha256:4e88556c32fdf47f64165a2180ba4552f8bb32c1103a2fafd05723a0bd42bd4b"}, + {file = "win32_setctime-1.1.0-py3-none-any.whl", hash = "sha256:231db239e959c2fe7eb1d7dc129f11172354f98361c4fa2d6d2d7e278baa8aad"}, + {file = "win32_setctime-1.1.0.tar.gz", hash = "sha256:15cf5750465118d6929ae4de4eb46e8edae9a5634350c01ba582df868e932cb2"}, ] zipp = [ - {file = "zipp-3.5.0-py3-none-any.whl", hash = "sha256:957cfda87797e389580cb8b9e3870841ca991e2125350677b2ca83a0e99390a3"}, - {file = "zipp-3.5.0.tar.gz", hash = "sha256:f5812b1e007e48cff63449a5e9f4e7ebea716b4111f9c4f9a645f91d579bf0c4"}, + {file = "zipp-3.7.0-py3-none-any.whl", hash = "sha256:b47250dd24f92b7dd6a0a8fc5244da14608f3ca90a5efcd37a3b1642fac9a375"}, + {file = "zipp-3.7.0.tar.gz", hash = "sha256:9f50f446828eb9d45b267433fd3e9da8d801f614129124863f9c51ebceafb87d"}, ] diff --git a/pyproject.toml b/pyproject.toml index b858a205..6bfb5769 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,86 +1,52 @@ [tool.poetry] name = "botx" -version = "0.28.0" -description = "A little python framework for building bots for eXpress" -license = "MIT" +version = "0.30.0" +description = "A python library for interacting with eXpress BotX API" authors = [ "Sidnev Nikolay ", "Maxim Gorbachev ", - "Alexander Samoylenko " + "Alexander Samoylenko ", + "Arseniy Zhiltsov " ] -readme = "README.md" documentation = "https://expressapp.github.io/pybotx" repository = "https://github.com/ExpressApp/pybotx" + [tool.poetry.dependencies] -python = "^3.7" +python = ">=3.8,<3.11" -base64io = "^1.0.3" -httpx = "^0.19.0" -loguru = "^0.5.0" -pydantic = "^1.0.0" -typing-extensions = { version = "^3.7.4", python = "<3.8" } +aiofiles = ">=0.7.0,<0.9.0" +httpx = ">=0.18.0,<0.22.0" +loguru = ">=0.6.0,<0.7.0" +mypy-extensions = ">=0.2.0,<0.5.0" +pydantic = ">=1.6.0,<1.9.0" +typing-extensions = ">=3.7.4,<5.0.0" -# for testing by users -aiofiles = { version = "^0.7.0", optional = true } -molten = { version = "^1.0.1", optional = true } -python-multipart = { version = "^0.0.5", optional = true } [tool.poetry.dev-dependencies] -# tasks -nox = "^2021.0.0" -# formatters -black = "^20.8b1" -isort = "^5.9" -autoflake = "^1.4" -add-trailing-comma = "^2.0.1" -# linters -mypy = "^0.812" -wemake-python-styleguide = "^0.15" -flake8-pytest-style = "^1.1.1" -# tests -pytest = "^6.0.0" -pytest-asyncio = "^0.15.0" -pytest-cov = "^2.8.1" -pytest-clarity = "^0.3.0-alpha.0" -coverage-conditional-plugin = "^0.4.0" -starlette = "^0.16.0" -# docs -mkdocs = "^1.1" -mkdocs-material = "^7.0.0" -markdown-include = "^0.6.0" -mkdocstrings = "^0.15.0" -livereload = "^2.6.3" +add-trailing-comma = "2.2.1" +autoflake = "1.4.0" +black = "21.12b0" +isort = "5.10.1" +mypy = "0.910.0" +wemake-python-styleguide = "0.16.0" +bandit = "1.7.2" # https://github.com/PyCQA/bandit/issues/837 + +pytest = "6.2.5" +pytest-asyncio = "0.16.0" +pytest-cov = "3.0.0" +python-dotenv = "0.19.2" +requests = "2.26.0" +respx = "0.19.0" -[tool.poetry.extras] -tests = ["aiofiles", "molten", "python-multipart"] +mkdocs = "1.2.3" +mkdocs-material = "8.1.9" +markdown = "3.3.6" # https://github.com/python-poetry/poetry/issues/4777 -[tool.black] -target_version = ['py37', 'py38'] -include = '\.pyi?$' -exclude = ''' -/(\.git/ - |\.eggs - |\.hg - |__pycache__ - |\.cache - |\.ipynb_checkpoints - |\.mypy_cache - |\.pytest_cache - |\.tox - |\.venv - |node_modules - |_build - |buck-out - |build - |dist - |media - |infrastructure - |templates - |locale -)/ -''' +fastapi = "0.73.0" +starlette = "0.17.1" # TODO: Drop dependency after updating end-to-end test +uvicorn = "0.16.0" [build-system] -requires = ["poetry>=0.1.0"] +requires = ["poetry>=0.12"] build-backend = "poetry.masonry.api" diff --git a/scripts/docs-format b/scripts/docs-format new file mode 100755 index 00000000..7acc651f --- /dev/null +++ b/scripts/docs-format @@ -0,0 +1,20 @@ +#!/usr/bin/env bash + +set -ex + +autoflake --recursive --in-place \ + --remove-all-unused-imports \ + --ignore-init-module-imports \ + docs/snippets +isort --profile black docs/snippets +black docs/snippets + +find docs/snippets -type f -name "*.py" | xargs add-trailing-comma --py36-plus --exit-zero-even-if-changed + +# This `black` is needed again in order to transfer parameters/arguments to new lines +# after inserting commas. +# The first `black` won't be able to transfer parameters/arguments to new lines because +# there is no comma at the end of the line. +# Inserting commas must be after the first `black`, so that there is one new line break, +# if the line is out of max-line-length. +black botx tests > /dev/null diff --git a/scripts/docs-lint b/scripts/docs-lint new file mode 100755 index 00000000..c73e32d6 --- /dev/null +++ b/scripts/docs-lint @@ -0,0 +1,9 @@ +#!/usr/bin/env bash + +set -ex + +black --check --diff docs/snippets +isort --profile black --check-only docs/snippets + +mypy docs/snippets +flake8 docs/snippets diff --git a/scripts/format b/scripts/format new file mode 100755 index 00000000..dd2b08e8 --- /dev/null +++ b/scripts/format @@ -0,0 +1,21 @@ +#!/usr/bin/env bash + +set -ex + +autoflake --recursive --in-place \ + --remove-all-unused-imports \ + --ignore-init-module-imports \ + botx tests +isort --profile black botx tests +black botx tests + +find botx -type f -name "*.py" | xargs add-trailing-comma --py36-plus --exit-zero-even-if-changed +find tests -type f -name "*.py" | xargs add-trailing-comma --py36-plus --exit-zero-even-if-changed + +# This `black` is needed again in order to transfer parameters/arguments to new lines +# after inserting commas. +# The first `black` won't be able to transfer parameters/arguments to new lines because +# there is no comma at the end of the line. +# Inserting commas must be after the first `black`, so that there is one new line break, +# if the line is out of max-line-length. +black botx tests > /dev/null diff --git a/scripts/lint b/scripts/lint new file mode 100755 index 00000000..c2da4b1f --- /dev/null +++ b/scripts/lint @@ -0,0 +1,11 @@ +#!/usr/bin/env bash + +set -ex + +black --check --diff botx tests +isort --profile black --check-only botx tests + +mypy botx tests +flake8 botx tests + +./scripts/wip_marks diff --git a/scripts/test b/scripts/test new file mode 100755 index 00000000..743b0a6b --- /dev/null +++ b/scripts/test @@ -0,0 +1,5 @@ +#!/usr/bin/env bash + +set -ex + +pytest ${@} diff --git a/scripts/wip_marks b/scripts/wip_marks new file mode 100755 index 00000000..daddf39e --- /dev/null +++ b/scripts/wip_marks @@ -0,0 +1,9 @@ +#!/usr/bin/env bash + +wip_markers=$(find . -name "*.py" -print0 | xargs -0 grep -n "@pytest.mark.wip") + +if [[ "$wip_markers" ]] +then + printf "Some wip marks found:\n%s\n" "$wip_markers" >&2 + exit 1 +fi diff --git a/setup.cfg b/setup.cfg index 7639e7a4..eed2b5d5 100644 --- a/setup.cfg +++ b/setup.cfg @@ -1,206 +1,153 @@ -# All configuration for plugins and other utils is defined here. -# Read more about `setup.cfg`: -# https://docs.python.org/3/distutils/configfile.html - - -[isort] -# isort configuration: -# https://github.com/timothycrosley/isort/wiki/isort-Settings -include_trailing_comma = true -# See https://github.com/timothycrosley/isort#multi-line-output-modes -multi_line_output = 3 -line_length = 88 -force_grid_wrap = 0 -combine_as_imports = True - - -[darglint] -# darglint configuration: -# https://github.com/terrencepreilly/darglint -strictness = long - - -[tool:pytest] -# Directories that are not visited by pytest collector: -norecursedirs = *.egg .eggs dist build docs .tox .git __pycache__ - -# You will need to measure your tests speed with `-n auto` and without it, -# so you can see whether it gives you any performance gain, or just gives -# you an overhead. See `docs/template/development-process.rst`. -addopts = - --strict-markers - --tb=short - --cov=botx - --cov=tests - --cov-branch - --cov-report=term-missing - --cov-report=html - --cov-report=xml - --no-cov-on-fail - --cov-fail-under=100 - - -[coverage:run] -# Here we specify plugins for coverage to be used: -plugins = - coverage_conditional_plugin - -omit = - - -[coverage:report] -precision = 2 -exclude_lines = - pragma: no cover - raise NotImplementedError - if TYPE_CHECKING: - except ImportError: - - -[coverage:coverage_conditional_plugin] -# Here we specify our pragma rules: -rules = - "sys_version_info < (3, 8)": py-lt-38 - - [mypy] -# Mypy configuration: -# https://mypy.readthedocs.io/en/latest/config_file.html plugins = pydantic.mypy +warn_unused_configs = True +disallow_any_generics = True +disallow_subclassing_any = True +disallow_untyped_calls = True disallow_untyped_defs = True -strict_optional = True +disallow_incomplete_defs = True +check_untyped_defs = True +disallow_untyped_decorators = True +no_implicit_optional = True warn_redundant_casts = True warn_unused_ignores = True +warn_return_any = True +no_implicit_reexport = True +strict_equality = True show_error_codes = True +[mypy-tests.*] +# https://github.com/python/mypy/issues/9689 +disallow_untyped_decorators = False -[pydantic-mypy] -init_forbid_extra = True -init_typed = True -warn_required_dynamic_aliases = True - +[mypy-aiofiles.*] +ignore_missing_imports = True -[mypy-noxfile] -# Nox decorators return untyped callables -disallow_untyped_decorators = false +[mypy-pytest.*] +ignore_missing_imports = True +[mypy-respx.*] +ignore_missing_imports = True -[mypy-tests.*] -# ignore mypy on tests package -ignore_errors = true +[isort] +profile = black +multi_line_output = 3 -[mypy-base64io.*] -ignore_missing_imports = True +[darglint] +# darglint configuration: +# https://github.com/terrencepreilly/darglint +strictness = short +docstring_style = sphinx [flake8] format = wemake -show-source = True -statistics = False +show-source = true -# Flake plugins: max-line-length = 88 inline-quotes = double -i-control-code = False -# currently 13 imports are used for Bot definition -max-imports = 13 -nested-classes-whitelist = Config -allowed-domain-names = - # handler is something similar to "views" from common framework, but for bot: - handler, - - # BotX API is built with similar to json-rpc approach and use "result" field for responses: - result, - - # file is field that is used in BotX API and it's an entity provided by library: - file, -pytest-raises-require-match-for = - -# Excluding some directories: -exclude = .git,__pycache__,.venv,.eggs,*.egg - -# Docs: https://github.com/snoack/flake8-per-file-ignores -# You can completely or partially disable our custom checks, -# to do so you have to ignore `WPS` letters for all python files: -per-file-ignores = - # WPS: - # "validated_values" is required name for pydantic validator in case it receives validated query_params: - botx/collecting/handlers/validators.py: WPS110, - - # re-exports from library using __all__: - botx/__init__.py: WPS201, WPS203, WPS235, WPS410 - - # allow inheritance from builtin, since there are enums for pydantic, module members, OverusedStringViolation: - botx/models/enums.py: WPS600, WPS202, WPS226 - - # TODO: simplify test utils - botx/testing/testing_client/base.py: WPS201 - botx/testing/botx_mock/asgi/routes/chats.py: WPS202, WPS204 - botx/testing/botx_mock/wsgi/routes/chats.py: WPS202, WPS204 - botx/testing/botx_mock/asgi/routes/stickers.py: WPS202, WPS204, WPS226 - botx/testing/botx_mock/wsgi/routes/stickers.py: WPS202, WPS204, WPS226 - - # magic method(__root__) - botx/testing/building/attachments.py: WPS609 - botx/testing/building/entites.py: WPS609 - - # E800 for disable formatter - botx/models/attachments.py: WPS110, WPS125, WPS202, E800 - - # allow noqa overuse and many imports (we should reconsider WPS ignores) - botx/models/messages/sending/message.py: WPS402, WPS201 +nested_classes_whitelist = Config +allowed_domain_names = data, handler, result, content, file - # cls to pydantic validators - botx/models/*.py: N805 +per-file-ignores = + botx/bot/bot.py:WPS203, + botx/constants.py:WPS432, + botx/__init__.py:WPS203,WPS410,WPS412,F401, + # https://github.com/wemake-services/wemake-python-styleguide/issues/2172 + botx/bot/handler_collector.py:WPS437, + botx/client/notifications_api/internal_bot_notification.py:WPS202, + # Complex model converting + botx/models/message/incoming_message.py:WPS232, + # WPS reacts at using `}` in f-strings + botx/models/message/mentions.py:WPS226, + # Protected attr usage is OK with async_files + botx/models/async_files.py:WPS437, + botx/models/api_base.py:WPS232,WPS231,WPS110,WPS440 + # This ignores make logger code cleaner + botx/logger.py:WPS219,WPS226 + + tests/*:DAR101,E501,WPS110,WPS114,WPS116,WPS118,WPS202,WPS221,WPS226,WPS237,WPS402,WPS420,WPS428,WPS430,WPS432,WPS441,WPS442,WPS520,PT011,S105,S106 - # disable most linting issues for tests: - # TODO: configure linting for tests more strictly - tests/*.py: D, S101, S106, WPS, B015 +ignore = + # This project uses google style docstring + RST, + # Upper-case constant in class + WPS115, + # Too many module members + WPS202, + # Too many arguments + WPS211, + # f-strings + WPS305, + # Class without base class + WPS306, + # Implicit string concatenation + WPS326, + # Explicit string concatenation + WPS336, + # Module docstring + D100, + # Class docstring + D101, + # Method docstring + D102, + # Function docstring + D103, + # Package docstring + D104, + # Magic method docstring + D105, + # Nested class docstring + D106, + # __init__ docstring + D107, + # Allow empty line after docstring + D202, + # Line break before binary operator + W503, + # Too many methods + WPS214, + # Too many imports + WPS201, + # Overused expression + WPS204, + # Too many local vars + WPS210, + # Too many imported names from module + WPS235, + # Multiple conditions + WPS337, + # Nested imports (often used with ImportError) + WPS433, + # Forbidden `@staticmethod` + WPS602, + # Forbidden `assert` + S101, - # module with content examplest - botx/testing/content.py: E501 - # imports into module with collector model is ok - botx/bots/bots.py: WPS201 +[tool:pytest] +testpaths = tests - # many chat methods - botx/bots/mixins/requests/chats.py: WPS201, WPS214 +addopts = + --strict-markers + --tb=short + --cov=botx + --cov-report=term-missing + --cov-branch + --no-cov-on-fail + --cov-fail-under=100 - botx/exceptions.py: WPS202 - botx/models/events.py: WPS202 +markers = + wip: "Work in progress" + mock_authorization: "Mock authorization" - # too many module members - # too many imported names from a module - # found overused expression - botx/bots/mixins/requests/mixin.py: WPS235 - botx/testing/botx_mock/asgi/application.py: WPS235 - botx/testing/botx_mock/asgi/routes/chats.py: WPS202,WPS204,WPS235 - botx/testing/botx_mock/wsgi/application.py: WPS235 - botx/testing/botx_mock/wsgi/routes/chats.py: WPS202,WPS204,WPS235 -# Disable some checks: -ignore = - # Docs: - # Disable nested classes documentation, since only Config for pydantic is allowed: - D106, - # This project uses google style and mkdocs for docs: - RST, - - # WPS: - # 3xx - # Disable required inheritance from object: - WPS306, - # Allow implicit string concatenation - WPS326, - - # 6xx - # A lot of functionality in this lib is build around async __call__: - WPS610, - - # TODO: - # WPS bugs: - WPS601, - - # Asserts in code is OK - S101, +[coverage:report] +exclude_lines = + pragma: no cover + if TYPE_CHECKING: + raise NotImplementedError + except ImportError: + ... # noqa: WPS428 + def __repr__ diff --git a/tests/test_bots/test_bots/test_decorators/__init__.py b/tests/client/__init__.py similarity index 100% rename from tests/test_bots/test_bots/test_decorators/__init__.py rename to tests/client/__init__.py diff --git a/tests/test_bots/test_mixins/__init__.py b/tests/client/bots_api/__init__.py similarity index 100% rename from tests/test_bots/test_mixins/__init__.py rename to tests/client/bots_api/__init__.py diff --git a/tests/client/bots_api/test_get_token.py b/tests/client/bots_api/test_get_token.py new file mode 100644 index 00000000..f371e84e --- /dev/null +++ b/tests/client/bots_api/test_get_token.py @@ -0,0 +1,78 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + InvalidBotAccountError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__get_token__invalid_bot_account_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_signature: str, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response(HTTPStatus.UNAUTHORIZED), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + with pytest.raises(InvalidBotAccountError) as exc: + async with lifespan_wrapper(built_bot) as bot: + await bot.get_token(bot_id=bot_id) + + # - Assert - + assert "failed with code 401" in str(exc.value) + assert endpoint.called + + +async def test__get_token__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_signature: str, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": "token", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + token = await bot.get_token(bot_id=bot_id) + + # - Assert - + assert token == "token" + assert endpoint.called diff --git a/tests/test_bots/test_mixins/test_requests/__init__.py b/tests/client/chats_api/__init__.py similarity index 100% rename from tests/test_bots/test_mixins/test_requests/__init__.py rename to tests/client/chats_api/__init__.py diff --git a/tests/client/chats_api/test_add_admin.py b/tests/client/chats_api/test_add_admin.py new file mode 100644 index 00000000..14fd58f9 --- /dev/null +++ b/tests/client/chats_api/test_add_admin.py @@ -0,0 +1,278 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + CantUpdatePersonalChatError, + ChatNotFoundError, + HandlerCollector, + InvalidBotXStatusCodeError, + InvalidUsersListError, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__promote_to_chat_admins__unexpected_bad_request_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "status": "error", + "reason": "some_reason", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidBotXStatusCodeError) as exc: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "some_reason" in str(exc.value) + assert endpoint.called + + +async def test__promote_to_chat_admins__cant_update_personal_chat_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "status": "error", + "reason": "chat_members_not_modifiable", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(CantUpdatePersonalChatError) as exc: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "chat_members_not_modifiable" in str(exc.value) + assert "Personal chat couldn't have admins" in str(exc.value) + assert endpoint.called + + +async def test__promote_to_chat_admins__invalid_users_list_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "status": "error", + "reason": "admins_not_changed", + "errors": ["Admins have not changed"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidUsersListError) as exc: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "admins_not_changed" in str(exc.value) + assert "Specified users are already admins or missing from chat" in str(exc.value) + assert endpoint.called + + +async def test__promote_to_chat_admins__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "no_permission_for_operation", + "errors": ["Sender is not chat admin"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "sender": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__promote_to_chat_admins__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": ["Chat not found"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__promote_to_chat_admins__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_admin", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": True}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.promote_to_chat_admins( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_add_user.py b/tests/client/chats_api/test_add_user.py new file mode 100644 index 00000000..eaaed589 --- /dev/null +++ b/tests/client/chats_api/test_add_user.py @@ -0,0 +1,145 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__add_users_to_chat__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": ["Chat not found"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.add_users_to_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__add_users_to_chat__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "no_permission_for_operation", + "errors": ["Sender is not chat admin"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "sender": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.add_users_to_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__add_users_to_chat__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/add_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": True}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.add_users_to_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_chat_info.py b/tests/client/chats_api/test_chat_info.py new file mode 100644 index 00000000..be9979f1 --- /dev/null +++ b/tests/client/chats_api/test_chat_info.py @@ -0,0 +1,142 @@ +from datetime import datetime as dt +from http import HTTPStatus +from typing import Callable +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatInfo, + ChatInfoMember, + ChatNotFoundError, + ChatTypes, + HandlerCollector, + UserKinds, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__chat_info__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/chats/info", + headers={"Authorization": "Bearer token"}, + params={"group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "error_description": "Chat with id dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4 not found", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.chat_info( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__chat_info__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + datetime_formatter: Callable[[str], dt], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/chats/info", + headers={"Authorization": "Bearer token"}, + params={"group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "chat_type": "group_chat", + "creator": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "description": None, + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "inserted_at": "2019-08-29T11:22:48.358586Z", + "members": [ + { + "admin": True, + "user_huid": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "user_kind": "user", + }, + { + "admin": False, + "user_huid": "705df263-6bfd-536a-9d51-13524afaab5c", + "user_kind": "botx", + }, + ], + "name": "Group Chat Example", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + chat_info = await bot.chat_info( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert chat_info == ChatInfo( + chat_type=ChatTypes.GROUP_CHAT, + creator_id=UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + description=None, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + created_at=datetime_formatter("2019-08-29T11:22:48.358586Z"), + members=[ + ChatInfoMember( + is_admin=True, + huid=UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + kind=UserKinds.RTS_USER, + ), + ChatInfoMember( + is_admin=False, + huid=UUID("705df263-6bfd-536a-9d51-13524afaab5c"), + kind=UserKinds.BOT, + ), + ], + name="Group Chat Example", + ) + + assert endpoint.called diff --git a/tests/client/chats_api/test_create_chat.py b/tests/client/chats_api/test_create_chat.py new file mode 100644 index 00000000..63d242a1 --- /dev/null +++ b/tests/client/chats_api/test_create_chat.py @@ -0,0 +1,159 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatCreationError, + ChatCreationProhibitedError, + ChatTypes, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__create_chat__bot_have_no_permissions_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/create", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "name": "Test chat name", + "description": None, + "chat_type": "group_chat", + "members": [], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "chat_creation_is_prohibited", + "errors": ["This bot is not allowed to create chats"], + "error_data": { + "bot_id": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatCreationProhibitedError) as exc: + await bot.create_chat( + bot_id=bot_id, + name="Test chat name", + chat_type=ChatTypes.GROUP_CHAT, + huids=[], + ) + + # - Assert - + assert endpoint.called + assert "chat_creation_is_prohibited" in str(exc.value) + + +async def test__create_chat__botx_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/create", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "name": "Test chat name", + "description": None, + "chat_type": "group_chat", + "members": [], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.UNPROCESSABLE_ENTITY, + json={ + "status": "error", + "reason": "|specified reason|", + "errors": ["|specified errors|"], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatCreationError) as exc: + await bot.create_chat( + bot_id=bot_id, + name="Test chat name", + chat_type=ChatTypes.GROUP_CHAT, + huids=[], + ) + + # - Assert - + assert endpoint.called + assert "specified reason" in str(exc.value) + + +async def test__create_chat__maximum_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/create", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "name": "Test chat name", + "description": "Test description", + "chat_type": "group_chat", + "members": ["2fc83441-366a-49ba-81fc-6c39f065bb58"], + "shared_history": True, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": {"chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + created_chat_id = await bot.create_chat( + bot_id=bot_id, + name="Test chat name", + chat_type=ChatTypes.GROUP_CHAT, + huids=[UUID("2fc83441-366a-49ba-81fc-6c39f065bb58")], + description="Test description", + shared_history=True, + ) + + # - Assert - + assert created_chat_id == UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa") + assert endpoint.called diff --git a/tests/client/chats_api/test_disable_stealth.py b/tests/client/chats_api/test_disable_stealth.py new file mode 100644 index 00000000..d8993c9a --- /dev/null +++ b/tests/client/chats_api/test_disable_stealth.py @@ -0,0 +1,133 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__disable_stealth__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_disable", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={"group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "no_permission_for_operation", + "errors": ["Sender is not chat admin"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "sender": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.disable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__disable_stealth__chat_not_found_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_disable", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={"group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": ["Chat not found"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.disable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__disable_stealth__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_disable", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={"group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": True}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.disable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_list_chats.py b/tests/client/chats_api/test_list_chats.py new file mode 100644 index 00000000..ad53b76c --- /dev/null +++ b/tests/client/chats_api/test_list_chats.py @@ -0,0 +1,81 @@ +from datetime import datetime +from http import HTTPStatus +from typing import Callable +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatListItem, + ChatTypes, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__list_chats__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + datetime_formatter: Callable[[str], datetime], +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/chats/list", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": [ + { + "group_chat_id": "740cf331-d833-5250-b5a5-5b5cbc697ff5", + "chat_type": "group_chat", + "name": "Chat Name", + "description": "Desc", + "members": [ + "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "705df263-6bfd-536a-9d51-13524afaab5c", + ], + "inserted_at": "2019-08-29T11:22:48.358586Z", + "updated_at": "2019-08-30T21:02:10.453786Z", + }, + ], + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + chats = await bot.list_chats(bot_id=bot_id) + + # - Assert - + assert chats == [ + ChatListItem( + chat_id=UUID("740cf331-d833-5250-b5a5-5b5cbc697ff5"), + chat_type=ChatTypes.GROUP_CHAT, + name="Chat Name", + description="Desc", + members=[ + UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + UUID("705df263-6bfd-536a-9d51-13524afaab5c"), + ], + created_at=datetime_formatter("2019-08-29T11:22:48.358586Z"), + updated_at=datetime_formatter("2019-08-30T21:02:10.453786Z"), + ), + ] + assert endpoint.called diff --git a/tests/client/chats_api/test_pin_message.py b/tests/client/chats_api/test_pin_message.py new file mode 100644 index 00000000..46cd40cc --- /dev/null +++ b/tests/client/chats_api/test_pin_message.py @@ -0,0 +1,147 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__pin_message__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/pin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "error_data": { + "bot_id": "f9e1c958-bf81-564e-bff2-a2943869af15", + "error_description": "Bot doesn't have permission for this operation in current chat", + "group_chat_id": "5680c26a-07a5-5b40-a6ff-f5e7e68fed25", + }, + "errors": [], + "reason": "no_permission_for_operation", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.pin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + sync_id=UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"), + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__pin_message__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/pin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "error_description": "Chat with specified id not found", + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.pin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + sync_id=UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"), + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__pin_message__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/pin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": "message_pinned"}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.pin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + sync_id=UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_remove_user.py b/tests/client/chats_api/test_remove_user.py new file mode 100644 index 00000000..e39906cc --- /dev/null +++ b/tests/client/chats_api/test_remove_user.py @@ -0,0 +1,145 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__remove_users_from_chat__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/remove_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "no_permission_for_operation", + "errors": ["Sender is not chat admin"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "sender": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.remove_users_from_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__remove_users_from_chat__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/remove_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": ["Chat not found"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.remove_users_from_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__remove_users_from_chat__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/remove_user", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "user_huids": ["f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": True}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.remove_users_from_chat( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + huids=[UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1")], + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_set_stealth.py b/tests/client/chats_api/test_set_stealth.py new file mode 100644 index 00000000..22addd5f --- /dev/null +++ b/tests/client/chats_api/test_set_stealth.py @@ -0,0 +1,148 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__enable_stealth__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_set", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "status": "error", + "reason": "no_permission_for_operation", + "errors": ["Sender is not chat admin"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + "sender": "a465f0f3-1354-491c-8f11-f400164295cb", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.enable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__enable_stealth__chat_not_found_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_set", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": ["Chat not found"], + "error_data": { + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.enable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__enable_stealth__maximum_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/stealth_set", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "disable_web": True, + "burn_in": 100, + "expire_in": 1000, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": True, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.enable_stealth( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + disable_web_client=True, + ttl_after_read=100, + total_ttl=1000, + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/chats_api/test_unpin_message.py b/tests/client/chats_api/test_unpin_message.py new file mode 100644 index 00000000..30429f90 --- /dev/null +++ b/tests/client/chats_api/test_unpin_message.py @@ -0,0 +1,141 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + PermissionDeniedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__unpin_message__permission_denied_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/unpin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.FORBIDDEN, + json={ + "error_data": { + "bot_id": "f9e1c958-bf81-564e-bff2-a2943869af15", + "error_description": "Bot doesn't have permission for this operation in current chat", + "group_chat_id": "5680c26a-07a5-5b40-a6ff-f5e7e68fed25", + }, + "errors": [], + "reason": "no_permission_for_operation", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(PermissionDeniedError) as exc: + await bot.unpin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "no_permission_for_operation" in str(exc.value) + assert endpoint.called + + +async def test__unpin_message__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/unpin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "error_description": "Chat with specified id not found", + "group_chat_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.unpin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__unpin_message__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/chats/unpin_message", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": "message_unpinned"}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.unpin_message( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/test_bots/test_mixins/test_sending/__init__.py b/tests/client/events_api/__init__.py similarity index 100% rename from tests/test_bots/test_mixins/test_sending/__init__.py rename to tests/client/events_api/__init__.py diff --git a/tests/client/events_api/test_edit_event.py b/tests/client/events_api/test_edit_event.py new file mode 100644 index 00000000..7f0e55a9 --- /dev/null +++ b/tests/client/events_api/test_edit_event.py @@ -0,0 +1,304 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + BubbleMarkup, + EditMessage, + HandlerCollector, + KeyboardMarkup, + Mention, + OutgoingAttachment, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__edit_message__minimal_edit_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/edit_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_command_result_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.edit_message( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + ) + + # - Assert - + assert endpoint.called + + +async def test__edit_message__maximum_edit_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/edit_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "payload": { + "body": "@{mention:f3e176d5-ff46-4b18-b260-25008338c06e}!", + "metadata": {"message": "metadata"}, + "bubble": [ + [ + { + "command": "/bubble-button", + "data": {}, + "label": "Bubble button", + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "data": {}, + "label": "Keyboard button", + "opts": {"silent": True}, + }, + ], + ], + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_command_result_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + bubbles = BubbleMarkup() + bubbles.add_button(command="/bubble-button", label="Bubble button") + + keyboard = KeyboardMarkup() + keyboard.add_button(command="/keyboard-button", label="Keyboard button") + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.edit_message( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body=f"{Mention.user(UUID('8f3abcc8-ba00-4c89-88e0-b786beb8ec24'))}!", + metadata={"message": "metadata"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + ) + + # - Assert - + assert endpoint.called + + +async def test__edit_message__clean_message_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/edit_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "payload": { + "body": "", + "metadata": {}, + "bubble": [], + "keyboard": [], + "mentions": [], + }, + "sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "file": None, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_command_result_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.edit_message( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body="", + metadata={}, + bubbles=BubbleMarkup(), + keyboard=KeyboardMarkup(), + file=None, + ) + + # - Assert - + assert endpoint.called + + +async def test__edit__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/edit_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "payload": { + "body": "@{mention:f3e176d5-ff46-4b18-b260-25008338c06e}!", + "metadata": {"message": "metadata"}, + "bubble": [ + [ + { + "command": "/bubble-button", + "data": {}, + "label": "Bubble button", + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "data": {}, + "label": "Keyboard button", + "opts": {"silent": True}, + }, + ], + ], + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_command_result_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + bubbles = BubbleMarkup() + bubbles.add_button(command="/bubble-button", label="Bubble button") + + keyboard = KeyboardMarkup() + keyboard.add_button(command="/keyboard-button", label="Keyboard button") + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + message = EditMessage( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body=f"{Mention.user(UUID('8f3abcc8-ba00-4c89-88e0-b786beb8ec24'))}!", + metadata={"message": "metadata"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.edit(message=message) + + # - Assert - + assert endpoint.called diff --git a/tests/client/events_api/test_message_status_event.py b/tests/client/events_api/test_message_status_event.py new file mode 100644 index 00000000..7bd5f2d5 --- /dev/null +++ b/tests/client/events_api/test_message_status_event.py @@ -0,0 +1,123 @@ +from datetime import datetime +from http import HTTPStatus +from typing import Callable +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + EventNotFoundError, + HandlerCollector, + MessageStatus, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__get_message_status__event_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/events/fe1f285c-073e-4231-b190-2959f28168cc/status", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "event_not_found", + "errors": [], + "error_data": {"sync_id": "fe1f285c-073e-4231-b190-2959f28168cc"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(EventNotFoundError) as exc: + await bot.get_message_status( + bot_id=bot_id, + sync_id=UUID("fe1f285c-073e-4231-b190-2959f28168cc"), + ) + + # - Assert - + assert "event_not_found" in str(exc.value) + assert endpoint.called + + +async def test__get_message_status__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + datetime_formatter: Callable[[str], datetime], +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/events/fe1f285c-073e-4231-b190-2959f28168cc/status", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": { + "group_chat_id": "740cf331-d833-5250-b5a5-5b5cbc697ff5", + "sent_to": ["32bb051e-cee9-5c5c-9c35-f213ec18d11e"], + "read_by": [ + { + "user_huid": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "read_at": "2019-08-29T11:22:48.358586Z", + }, + ], + "received_by": [ + { + "user_huid": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "received_at": "2019-08-29T11:22:48.358586Z", + }, + ], + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + message_status = await bot.get_message_status( + bot_id=bot_id, + sync_id=UUID("fe1f285c-073e-4231-b190-2959f28168cc"), + ) + + # - Assert - + assert message_status == MessageStatus( + group_chat_id=UUID("740cf331-d833-5250-b5a5-5b5cbc697ff5"), + sent_to=[UUID("32bb051e-cee9-5c5c-9c35-f213ec18d11e")], + read_by={ + UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"): datetime_formatter( + "2019-08-29T11:22:48.358586Z", + ), + }, + received_by={ + UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"): datetime_formatter( + "2019-08-29T11:22:48.358586Z", + ), + }, + ) + assert endpoint.called diff --git a/tests/client/events_api/test_reply_event.py b/tests/client/events_api/test_reply_event.py new file mode 100644 index 00000000..5d1cb8af --- /dev/null +++ b/tests/client/events_api/test_reply_event.py @@ -0,0 +1,289 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + BubbleMarkup, + HandlerCollector, + KeyboardMarkup, + Mention, + OutgoingAttachment, + ReplyMessage, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__reply_message__minimal_filled_reply_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/reply_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "source_sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "reply": { + "status": "ok", + "body": "Replied", + }, + "opts": {"raw_mentions": True}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_reply_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.reply_message( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body="Replied", + ) + + # - Assert - + assert endpoint.called + + +async def test__reply_message__maximum_filled_reply_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/reply_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "source_sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "reply": { + "status": "ok", + "body": "@{mention:f3e176d5-ff46-4b18-b260-25008338c06e}!", + "metadata": {"message": "metadata"}, + "opts": { + "buttons_auto_adjust": True, + "silent_response": True, + }, + "bubble": [ + [ + { + "command": "/bubble-button", + "label": "Bubble button", + "data": {}, + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "label": "Keyboard button", + "data": {}, + "opts": {"silent": True}, + }, + ], + ], + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + "opts": { + "raw_mentions": True, + "stealth_mode": True, + "notification_opts": {"send": True, "force_dnd": True}, + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_reply_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + bubbles = BubbleMarkup() + bubbles.add_button(command="/bubble-button", label="Bubble button") + + keyboard = KeyboardMarkup() + keyboard.add_button(command="/keyboard-button", label="Keyboard button") + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.reply_message( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body=f"{Mention.user(UUID('8f3abcc8-ba00-4c89-88e0-b786beb8ec24'))}!", + metadata={"message": "metadata"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + silent_response=True, + markup_auto_adjust=True, + stealth_mode=True, + send_push=True, + ignore_mute=True, + ) + + # - Assert - + assert endpoint.called + + +async def test__reply__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/reply_event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "source_sync_id": "8ba66c5b-40bf-5c77-911d-519cb4e382e9", + "reply": { + "status": "ok", + "body": "@{mention:f3e176d5-ff46-4b18-b260-25008338c06e}!", + "metadata": {"message": "metadata"}, + "opts": { + "buttons_auto_adjust": True, + "silent_response": True, + }, + "bubble": [ + [ + { + "command": "/bubble-button", + "label": "Bubble button", + "data": {}, + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "label": "Keyboard button", + "data": {}, + "opts": {"silent": True}, + }, + ], + ], + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + "opts": { + "raw_mentions": True, + "stealth_mode": True, + "notification_opts": {"send": True, "force_dnd": True}, + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "bot_reply_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + bubbles = BubbleMarkup() + bubbles.add_button(command="/bubble-button", label="Bubble button") + + keyboard = KeyboardMarkup() + keyboard.add_button(command="/keyboard-button", label="Keyboard button") + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + message = ReplyMessage( + bot_id=bot_id, + sync_id=UUID("8ba66c5b-40bf-5c77-911d-519cb4e382e9"), + body=f"{Mention.user(UUID('8f3abcc8-ba00-4c89-88e0-b786beb8ec24'))}!", + metadata={"message": "metadata"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + silent_response=True, + markup_auto_adjust=True, + stealth_mode=True, + send_push=True, + ignore_mute=True, + ) + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.reply(message=message) + + # - Assert - + assert endpoint.called diff --git a/tests/client/events_api/test_stop_typing_event.py b/tests/client/events_api/test_stop_typing_event.py new file mode 100644 index 00000000..6568eec4 --- /dev/null +++ b/tests/client/events_api/test_stop_typing_event.py @@ -0,0 +1,47 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, lifespan_wrapper + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__stop_typing__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/stop_typing", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": "stop_typing_event_pushed"}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.stop_typing( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/events_api/test_typing_event.py b/tests/client/events_api/test_typing_event.py new file mode 100644 index 00000000..31b6a00e --- /dev/null +++ b/tests/client/events_api/test_typing_event.py @@ -0,0 +1,47 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, lifespan_wrapper + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__start_typing__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/events/typing", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok", "result": "typing_event_pushed"}, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.start_typing( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/test_clients/__init__.py b/tests/client/files_api/__init__.py similarity index 100% rename from tests/test_clients/__init__.py rename to tests/client/files_api/__init__.py diff --git a/tests/client/files_api/test_download_file.py b/tests/client/files_api/test_download_file.py new file mode 100644 index 00000000..9a68aa65 --- /dev/null +++ b/tests/client/files_api/test_download_file.py @@ -0,0 +1,251 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + FileDeletedError, + FileMetadataNotFound, + HandlerCollector, + InvalidBotXStatusCodeError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__download_file__unexpected_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "anything_not_found", + "errors": [], + "error_data": { + "group_chat_id": "84a12e71-3efc-5c34-87d5-84e3d9ad64fd", + "error_description": "42", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidBotXStatusCodeError) as exc: + await bot.download_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file_id=UUID("c3b9def2-b2c8-4732-b61f-99b9b110fa80"), + async_buffer=async_buffer, + ) + + # - Assert - + assert "anything_not_found" in str(exc.value) + assert endpoint.called + + +async def test__download_file__file_metadata_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "file_metadata_not_found", + "errors": [], + "error_data": { + "file_id": "e48c5612-b94f-4264-adc2-1bc36445a226", + "group_chat_id": "84a12e71-3efc-5c34-87d5-84e3d9ad64fd", + "error_description": "File with specified file_id and group_chat_id not found in file service", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(FileMetadataNotFound) as exc: + await bot.download_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file_id=UUID("c3b9def2-b2c8-4732-b61f-99b9b110fa80"), + async_buffer=async_buffer, + ) + + # - Assert - + assert "file_metadata_not_found" in str(exc.value) + assert endpoint.called + + +async def test__download_file__file_deleted_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NO_CONTENT, + json={ + "status": "error", + "reason": "file_deleted", + "errors": [], + "error_data": { + "link": "/example/file.jpeg", + "error_description": "File at the specified link has been deleted", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(FileDeletedError) as exc: + await bot.download_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file_id=UUID("c3b9def2-b2c8-4732-b61f-99b9b110fa80"), + async_buffer=async_buffer, + ) + + # - Assert - + assert "file_deleted" in str(exc.value) + assert endpoint.called + + +async def test__download_file__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "84a12e71-3efc-5c34-87d5-84e3d9ad64fd", + "error_description": "Chat with id 84a12e71-3efc-5c34-87d5-84e3d9ad64fd not found", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.download_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file_id=UUID("c3b9def2-b2c8-4732-b61f-99b9b110fa80"), + async_buffer=async_buffer, + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__download_file__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + content=b"Hello, world!\n", + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.download_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file_id=UUID("c3b9def2-b2c8-4732-b61f-99b9b110fa80"), + async_buffer=async_buffer, + ) + + # - Assert - + assert await async_buffer.read() == b"Hello, world!\n" + assert endpoint.called diff --git a/tests/client/files_api/test_upload_file.py b/tests/client/files_api/test_upload_file.py new file mode 100644 index 00000000..17df5dae --- /dev/null +++ b/tests/client/files_api/test_upload_file.py @@ -0,0 +1,125 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + ChatNotFoundError, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__download_file__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/files/upload", + # TODO: check data too, when files pattern will be ready + # https://github.com/lundberg/respx/issues/115 + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "84a12e71-3efc-5c34-87d5-84e3d9ad64fd", + "error_description": "Chat with id 84a12e71-3efc-5c34-87d5-84e3d9ad64fd not found", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ChatNotFoundError) as exc: + await bot.upload_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + async_buffer=async_buffer, + filename="test.txt", + ) + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__download_file__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/files/upload", + # TODO: check data too, when files pattern will be ready + # https://github.com/lundberg/respx/issues/115 + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "caption": "текст", + "duration": None, + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.upload_file( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + async_buffer=async_buffer, + filename="test.txt", + ) + + # - Assert - + assert endpoint.called diff --git a/tests/test_clients/test_clients/__init__.py b/tests/client/notifications_api/__init__.py similarity index 100% rename from tests/test_clients/test_clients/__init__.py rename to tests/client/notifications_api/__init__.py diff --git a/tests/client/notifications_api/test_direct_notification.py b/tests/client/notifications_api/test_direct_notification.py new file mode 100644 index 00000000..4e90d9df --- /dev/null +++ b/tests/client/notifications_api/test_direct_notification.py @@ -0,0 +1,974 @@ +import asyncio +from http import HTTPStatus +from typing import Any, Callable, Dict +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + AnswerDestinationLookupError, + Bot, + BotAccountWithSecret, + BotIsNotChatMemberError, + BubbleMarkup, + ChatNotFoundError, + FinalRecipientsListEmptyError, + HandlerCollector, + IncomingMessage, + KeyboardMarkup, + Mention, + OutgoingAttachment, + OutgoingMessage, + StealthModeDisabledError, + UnknownBotAccountError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__send_message__succeed( + respx_mock: MockRouter, + host: str, + bot_account: BotAccountWithSecret, + bot_id: UUID, + api_incoming_message_factory: Callable[..., Dict[str, Any]], +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "opts": { + "silent_response": True, + "buttons_auto_adjust": True, + }, + "status": "ok", + "body": "Hi!", + "metadata": {"foo": "bar"}, + "bubble": [ + [ + { + "command": "/bubble-button", + "data": {}, + "label": "Bubble button", + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "data": {}, + "label": "Keyboard button", + "opts": {"silent": True}, + }, + ], + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + "recipients": ["0a462a79-d9a2-4fad-8a96-7074f59daba9"], + "opts": { + "stealth_mode": True, + "notification_opts": { + "send": True, + "force_dnd": True, + }, + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + payload = api_incoming_message_factory( + bot_id=bot_id, + host=host, + group_chat_id="054af49e-5e18-4dca-ad73-4f96b6de63fa", + ) + + bubbles = BubbleMarkup() + bubbles.add_button( + command="/bubble-button", + label="Bubble button", + ) + + keyboard = KeyboardMarkup() + keyboard.add_button( + command="/keyboard-button", + label="Keyboard button", + ) + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + collector = HandlerCollector() + + outgoing_message = OutgoingMessage( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + body="Hi!", + metadata={"foo": "bar"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + silent_response=True, + markup_auto_adjust=True, + recipients=[UUID("0a462a79-d9a2-4fad-8a96-7074f59daba9")], + stealth_mode=True, + send_push=True, + ignore_mute=True, + ) + + @collector.command("/hello", description="Hello command") + async def hello_handler(message: IncomingMessage, bot: Bot) -> None: + await bot.send(message=outgoing_message) + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert endpoint.called + + +async def test__answer_message__no_incoming_message_error_raised( + host: str, + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(AnswerDestinationLookupError) as exc: + await bot.answer_message("Hi!") + + # - Assert - + assert "No IncomingMessage received" in str(exc.value) + + +async def test__answer_message__succeed( + respx_mock: MockRouter, + host: str, + bot_account: BotAccountWithSecret, + bot_id: UUID, + api_incoming_message_factory: Callable[..., Dict[str, Any]], +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "status": "ok", + "body": "Hi!", + "metadata": {"foo": "bar"}, + "bubble": [ + [ + { + "command": "/bubble-button", + "data": {}, + "label": "Bubble button", + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "data": {}, + "label": "Keyboard button", + "opts": {"silent": True}, + }, + ], + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + payload = api_incoming_message_factory( + bot_id=bot_id, + host=host, + group_chat_id="054af49e-5e18-4dca-ad73-4f96b6de63fa", + ) + + bubbles = BubbleMarkup() + bubbles.add_button( + command="/bubble-button", + label="Bubble button", + ) + + keyboard = KeyboardMarkup() + keyboard.add_button( + command="/keyboard-button", + label="Keyboard button", + ) + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + collector = HandlerCollector() + + @collector.command("/hello", description="Hello command") + async def hello_handler(message: IncomingMessage, bot: Bot) -> None: + await bot.answer_message( + "Hi!", + metadata={"foo": "bar"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + ) + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert endpoint.called + + +async def test__send_message__unknown_bot_account_error_raised( + respx_mock: MockRouter, + host: str, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + unknown_bot_id = UUID("51550ccc-dfd1-4d22-9b6f-a330145192b0") + direct_notification_endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UnknownBotAccountError) as exc: + await bot.send_message( + body="Hi!", + bot_id=unknown_bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert not direct_notification_endpoint.called + assert str(unknown_bot_id) in str(exc.value) + + +async def test__send_message__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "error_description": "Chat with specified id not found", + }, + }, + ) + + # - Assert - + with pytest.raises(ChatNotFoundError) as exc: + await task + + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__send_message__bot_is_not_a_chat_member_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "bot_is_not_a_chat_member", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "bot_id": "b165f00f-3154-412c-7f11-c120164257da", + "error_description": "Bot is not a chat member", + }, + }, + ) + + # - Assert - + with pytest.raises(BotIsNotChatMemberError) as exc: + await task + + assert "bot_is_not_a_chat_member" in str(exc.value) + assert endpoint.called + + +async def test__send_message__event_recipients_list_is_empty_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "event_recipients_list_is_empty", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "bot_id": "b165f00f-3154-412c-7f11-c120164257da", + "recipients_param": ["b165f00f-3154-412c-7f11-c120164257da"], + "error_description": "Event recipients list is empty", + }, + }, + ) + + # - Assert - + with pytest.raises(FinalRecipientsListEmptyError) as exc: + await task + + assert "event_recipients_list_is_empty" in str(exc.value) + assert endpoint.called + + +async def test__send_message__stealth_mode_disabled_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "stealth_mode_disabled", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "bot_id": "b165f00f-3154-412c-7f11-c120164257da", + "error_description": "Stealth mode disabled in specified chat", + }, + }, + ) + + # - Assert - + with pytest.raises(StealthModeDisabledError) as exc: + await task + + assert "stealth_mode_disabled" in str(exc.value) + assert endpoint.called + + +async def test__send_message__miminally_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__send_message__maximum_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + + body = f"Hi, {Mention.user(UUID('8f3abcc8-ba00-4c89-88e0-b786beb8ec24'))}!" + formatted_body = "Hi, @{mention:f3e176d5-ff46-4b18-b260-25008338c06e}!" + + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "opts": { + "silent_response": True, + "buttons_auto_adjust": True, + }, + "status": "ok", + "body": formatted_body, + "metadata": {"foo": "bar"}, + "bubble": [ + [ + { + "command": "/bubble-button", + "label": "Bubble button", + "data": {"foo": "bar"}, + "opts": { + "silent": False, + "h_size": 1, + "alert_text": "Alert text 1", + "show_alert": True, + "handler": "client", + }, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "label": "Keyboard button", + "data": {"baz": "quux"}, + "opts": { + "silent": True, + "h_size": 2, + "alert_text": "Alert text 2", + "show_alert": True, + }, + }, + ], + ], + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + ], + }, + "file": { + "file_name": "test.txt", + "data": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + }, + "recipients": ["41af5a7b-04c1-465e-8383-e3b1d9e76126"], + "opts": { + "stealth_mode": True, + "notification_opts": { + "send": True, + "force_dnd": True, + }, + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer(async_buffer, "test.txt") + + bubbles = BubbleMarkup() + bubbles.add_button( + command="/bubble-button", + label="Bubble button", + data={"foo": "bar"}, + silent=False, + width_ratio=1, + alert="Alert text 1", + process_on_client=True, + ) + + keyboard = KeyboardMarkup() + keyboard.add_button( + command="/keyboard-button", + label="Keyboard button", + data={"baz": "quux"}, + silent=True, + width_ratio=2, + alert="Alert text 2", + process_on_client=False, + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body=body, + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + metadata={"foo": "bar"}, + bubbles=bubbles, + keyboard=keyboard, + file=file, + silent_response=True, + markup_auto_adjust=True, + recipients=[UUID("41af5a7b-04c1-465e-8383-e3b1d9e76126")], + stealth_mode=True, + send_push=True, + ignore_mute=True, + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__send_message__all_mentions_types_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr( + "botx.models.message.mentions.uuid4", + lambda: UUID("f3e176d5-ff46-4b18-b260-25008338c06e"), + ) + + mentioned_user_huid = UUID("8f3abcc8-ba00-4c89-88e0-b786beb8ec24") + user_mention = Mention.user(mentioned_user_huid) + mentioned_contact_huid = UUID("1e0529fd-f091-4be9-93cc-6704a8957432") + contact_mention = Mention.contact(mentioned_contact_huid) + mentioned_chat_huid = UUID("454d73ad-1d32-4939-a708-e14b77414e86") + chat_mention = Mention.chat(mentioned_chat_huid, "Our chat") + mentioned_channel_huid = UUID("78198bec-3285-48d0-9fe2-c0eb3afaffd7") + channel_mention = Mention.channel(mentioned_channel_huid) + all_mention = Mention.all() + + body = ( + f"Hi, {user_mention}, want you to know, " + f"that I and {contact_mention} are getting married in a week. " + f"Here's a chat for all the invitees: {chat_mention}. " + f"And here is the news channel just in case: {channel_mention}. " + "In case of something incredible, " + f"I will notify you with {all_mention}, so you won't miss it." + ) + + formatted_body = ( + "Hi, @{mention:f3e176d5-ff46-4b18-b260-25008338c06e}, want you to know, " + "that I and @@{mention:f3e176d5-ff46-4b18-b260-25008338c06e} are getting married in a week. " + "Here's a chat for all the invitees: ##{mention:f3e176d5-ff46-4b18-b260-25008338c06e}. " + "And here is the news channel just in case: ##{mention:f3e176d5-ff46-4b18-b260-25008338c06e}. " + "In case of something incredible, " + "I will notify you with @{mention:f3e176d5-ff46-4b18-b260-25008338c06e}, so you won't miss it." + ) + + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "status": "ok", + "body": formatted_body, + "mentions": [ + { + "mention_type": "user", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "8f3abcc8-ba00-4c89-88e0-b786beb8ec24", + }, + }, + { + "mention_type": "contact", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "user_huid": "1e0529fd-f091-4be9-93cc-6704a8957432", + }, + }, + { + "mention_type": "chat", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "group_chat_id": "454d73ad-1d32-4939-a708-e14b77414e86", + "name": "Our chat", + }, + }, + { + "mention_type": "channel", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + "mention_data": { + "group_chat_id": "78198bec-3285-48d0-9fe2-c0eb3afaffd7", + }, + }, + { + "mention_type": "all", + "mention_id": "f3e176d5-ff46-4b18-b260-25008338c06e", + }, + ], + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body=body, + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__send_message__message_body_max_length_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + too_long_body = "1" * 4097 + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ValueError) as exc: + await bot.send_message( + body=too_long_body, + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ) + + # - Assert - + assert "Message body length exceeds 4096 symbols" in str(exc.value) + assert not endpoint.called + + +async def test__send_message__message_body_max_length_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + max_long_body = "1" * 4096 + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body=max_long_body, + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called diff --git a/tests/client/notifications_api/test_internal_bot_notification.py b/tests/client/notifications_api/test_internal_bot_notification.py new file mode 100644 index 00000000..cfb18357 --- /dev/null +++ b/tests/client/notifications_api/test_internal_bot_notification.py @@ -0,0 +1,297 @@ +import asyncio +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + BotIsNotChatMemberError, + ChatNotFoundError, + FinalRecipientsListEmptyError, + HandlerCollector, + RateLimitReachedError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__send_internal_bot_notification__rate_limit_reached_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/internal", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"foo": "bar"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.TOO_MANY_REQUESTS, + json={ + "status": "error", + "reason": "too_many_requests", + "errors": [], + "error_data": { + "bot_id": "b165f00f-3154-412c-7f11-c120164257da", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(RateLimitReachedError) as exc: + await bot.send_internal_bot_notification( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"foo": "bar"}, + ) + + # - Assert - + assert "too_many_requests" in str(exc.value) + assert endpoint.called + + +async def test__send_internal_bot_notification__chat_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/internal", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"foo": "bar"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_internal_bot_notification( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"foo": "bar"}, + ), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "error_description": ( + "Chat with id 21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3 not found" + ), + }, + }, + ) + + with pytest.raises(ChatNotFoundError) as exc: + await task + + # - Assert - + assert "chat_not_found" in str(exc.value) + assert endpoint.called + + +async def test__send_internal_bot_notification__bot_is_not_chat_member_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/internal", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"foo": "bar"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_internal_bot_notification( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"foo": "bar"}, + ), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "bot_is_not_a_chat_member", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "bot_id": str(bot_id), + "error_description": "Bot is not a chat member", + }, + }, + ) + + with pytest.raises(BotIsNotChatMemberError) as exc: + await task + + # - Assert - + assert "bot_is_not_a_chat_member" in str(exc.value) + assert endpoint.called + + +async def test__send_internal_bot_notification__final_recipients_list_empty_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/internal", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"foo": "bar"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_internal_bot_notification( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"foo": "bar"}, + ), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "event_recipients_list_is_empty", + "errors": [], + "error_data": { + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "bot_id": str(bot_id), + "recipients_param": ["b165f00f-3154-412c-7f11-c120164257da"], + "error_description": "Event recipients list is empty", + }, + }, + ) + + with pytest.raises(FinalRecipientsListEmptyError) as exc: + await task + + # - Assert - + assert "event_recipients_list_is_empty" in str(exc.value) + assert endpoint.called + + +async def test__send_internal_bot_notification__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/internal", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"foo": "bar"}, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_internal_bot_notification( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"foo": "bar"}, + ), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert await task == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called diff --git a/tests/client/notifications_api/test_markup.py b/tests/client/notifications_api/test_markup.py new file mode 100644 index 00000000..3421381b --- /dev/null +++ b/tests/client/notifications_api/test_markup.py @@ -0,0 +1,239 @@ +import asyncio +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + BubbleMarkup, + Button, + HandlerCollector, + KeyboardMarkup, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__markup__defaults_filled( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "status": "ok", + "body": "Hi!", + "bubble": [ + [ + { + "command": "/bubble-button", + "data": {}, + "label": "Bubble button", + "opts": {"silent": True}, + }, + ], + ], + "keyboard": [ + [ + { + "command": "/keyboard-button", + "data": {}, + "label": "Keyboard button", + "opts": {"silent": True}, + }, + ], + ], + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + bubbles = BubbleMarkup() + bubbles.add_button( + command="/bubble-button", + label="Bubble button", + ) + + keyboard = KeyboardMarkup() + keyboard.add_button( + command="/keyboard-button", + label="Keyboard button", + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + bubbles=bubbles, + keyboard=keyboard, + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__markup__correctly_built( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": { + "status": "ok", + "body": "Hi!", + "bubble": [ + [ + { + "command": "/bubble-button-1", + "data": {}, + "label": "Bubble button 1", + "opts": {"silent": True}, + }, + { + "command": "/bubble-button-2", + "data": {}, + "label": "Bubble button 2", + "opts": {"silent": True}, + }, + ], + [ + { + "command": "/bubble-button-3", + "data": {}, + "label": "Bubble button 3", + "opts": {"silent": True}, + }, + ], + [ + { + "command": "/bubble-button-4", + "data": {}, + "label": "Bubble button 4", + "opts": {"silent": True}, + }, + { + "command": "/bubble-button-5", + "data": {}, + "label": "Bubble button 5", + "opts": {"silent": True}, + }, + ], + ], + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + bubbles = BubbleMarkup() + + bubbles.add_button( + command="/bubble-button-1", + label="Bubble button 1", + new_row=False, + ) + bubbles.add_button( + command="/bubble-button-2", + label="Bubble button 2", + new_row=False, + ) + bubbles.add_button( + command="/bubble-button-3", + label="Bubble button 3", + ) + + button_4 = Button( + command="/bubble-button-4", + label="Bubble button 4", + ) + button_5 = Button( + command="/bubble-button-5", + label="Bubble button 5", + ) + bubbles.add_row([button_4, button_5]) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + bubbles=bubbles, + ), + ) + + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert (await task) == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +def test__markup__comparison() -> None: + # - Arrange - + button = Button("/test", "test") + + # - Assert - + assert BubbleMarkup([[button]]) == BubbleMarkup([[button]]) diff --git a/tests/test_clients/test_clients/test_async_client/__init__.py b/tests/client/smartapps_api/__init__.py similarity index 100% rename from tests/test_clients/test_clients/test_async_client/__init__.py rename to tests/client/smartapps_api/__init__.py diff --git a/tests/client/smartapps_api/test_smartapp_event.py b/tests/client/smartapps_api/test_smartapp_event.py new file mode 100644 index 00000000..67b4126c --- /dev/null +++ b/tests/client/smartapps_api/test_smartapp_event.py @@ -0,0 +1,187 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, lifespan_wrapper +from botx.models.async_files import Document, Image, Video, Voice +from botx.models.enums import AttachmentTypes + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__send_smartapp_event__miminally_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/smartapps/event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "ref": "921763b3-77e8-4f37-b97e-20f4517949b8", + "smartapp_id": str(bot_id), + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"key": "value"}, + "opts": {}, + "smartapp_api_version": 1, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "smartapp_event_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.send_smartapp_event( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + data={"key": "value"}, + ref=UUID("921763b3-77e8-4f37-b97e-20f4517949b8"), + ) + + # - Assert - + assert endpoint.called + + +async def test__send_smartapp_event__maximum_filled_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/smartapps/event", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "ref": "921763b3-77e8-4f37-b97e-20f4517949b8", + "smartapp_id": str(bot_id), + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "data": {"key": "value"}, + "opts": {"option": True}, + "smartapp_api_version": 1, + "async_files": [ + { + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + { + "type": "video", + "file": "https://link.to/file", + "file_mime_type": "video/mp4", + "file_name": "pass.mp4", + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "duration": 10, + }, + { + "type": "document", + "file": "https://link.to/file", + "file_mime_type": "plain/text", + "file_name": "pass.txt", + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + { + "type": "voice", + "file": "https://link.to/file", + "file_mime_type": "audio/mp3", + "file_name": "pass.mp3", + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "duration": 10, + }, + ], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "smartapp_event_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.send_smartapp_event( + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + ref=UUID("921763b3-77e8-4f37-b97e-20f4517949b8"), + data={"key": "value"}, + opts={"option": True}, + files=[ + Image( + type=AttachmentTypes.IMAGE, + filename="pass.png", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="image/png", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + Video( + type=AttachmentTypes.VIDEO, + filename="pass.mp4", + size=1502345, + is_async_file=True, + duration=10, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="video/mp4", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + Document( + type=AttachmentTypes.DOCUMENT, + filename="pass.txt", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="plain/text", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + Voice( + type=AttachmentTypes.VOICE, + filename="pass.mp3", + size=1502345, + is_async_file=True, + duration=10, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="audio/mp3", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ], + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/smartapps_api/test_smartapp_notification.py b/tests/client/smartapps_api/test_smartapp_notification.py new file mode 100644 index 00000000..aaf5ea32 --- /dev/null +++ b/tests/client/smartapps_api/test_smartapp_notification.py @@ -0,0 +1,55 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, lifespan_wrapper + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__send_smartapp_notification__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/smartapps/notification", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "705df263-6bfd-536a-9d51-13524afaab5c", + "smartapp_counter": 42, + "opts": {"message": "ping"}, + "smartapp_api_version": 1, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "smartapp_notification_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.send_smartapp_notification( + bot_id=bot_id, + chat_id=UUID("705df263-6bfd-536a-9d51-13524afaab5c"), + smartapp_counter=42, + opts={"message": "ping"}, + ) + + # - Assert - + assert endpoint.called diff --git a/tests/test_clients/test_clients/test_sync_client/__init__.py b/tests/client/stickers_api/__init__.py similarity index 100% rename from tests/test_clients/test_clients/test_sync_client/__init__.py rename to tests/client/stickers_api/__init__.py diff --git a/tests/client/stickers_api/test_add_sticker.py b/tests/client/stickers_api/test_add_sticker.py new file mode 100644 index 00000000..c8ea1e52 --- /dev/null +++ b/tests/client/stickers_api/test_add_sticker.py @@ -0,0 +1,318 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + InvalidBotXStatusCodeError, + InvalidEmojiError, + InvalidImageError, + Sticker, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) + +PNG_IMAGE = ( + b"\x89PNG\r\n\x1a\n" + b"\x00\x00\x00\rIHDR\x00\x00\x00\x01\x00\x00\x00\x01\x01\x03\x00" + b"\x00\x00%\xdbV\xca\x00\x00\x00\x03PLTE\x00\x00\x00\xa7z=\xda\x00" + b"\x00\x00\x01tRNS\x00@\xe6\xd8f\x00\x00\x00\nIDAT\x08\xd7c`\x00" + b"\x00\x00\x02\x00\x01\xe2!\xbc3\x00\x00\x00\x00IEND\xaeB`\x82" +) +PNG_IMAGE_B64 = ( + "iVBORw0KGgoAAAANSUhEUgAAAAEAAAABAQMAAAAl21bKAAAAA1BMVEUAAACnej3a" + "AAAAAXRSTlMAQObYZgAAAApJREFUCNdjYAAAAAIAAeIhvDMAAAAASUVORK5CYII=" +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__add_sticker__is_not_png_error_raised( + respx_mock: MockRouter, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(b"Hello, world!\n") + await async_buffer.seek(0) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ValueError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "Passed file is not PNG" in str(exc.value) + + +async def test__add_sticker__bad_file_size_error_raised( + respx_mock: MockRouter, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE + b"\x00" * (512 * 1024 + 1)) + await async_buffer.seek(0) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ValueError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "Passed file size is greater than 0.5 Mb" in str(exc.value) + + +async def test__add_sticker__unexpected_bad_request_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE) + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/stickers", + headers={"Authorization": "Bearer token"}, + json={"emoji": "🤔", "image": f"data:image/png;base64,{PNG_IMAGE_B64}"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "status": "error", + "reason": "some_reason", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidBotXStatusCodeError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "some_reason" in str(exc.value) + assert endpoint.called + + +async def test__add_sticker__sticker_pack_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE) + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/stickers", + headers={"Authorization": "Bearer token"}, + json={"emoji": "🤔", "image": f"data:image/png;base64,{PNG_IMAGE_B64}"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "error_data": {"pack_id": "26080153-a57d-5a8c-af0e-fdecee3c4435"}, + "errors": ["Failed to add sticker because pack not found."], + "reason": "pack_not_found", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "pack_not_found" in str(exc.value) + assert endpoint.called + + +async def test__add_sticker__invalid_emoji_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE) + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/stickers", + headers={"Authorization": "Bearer token"}, + json={"emoji": "🤔", "image": f"data:image/png;base64,{PNG_IMAGE_B64}"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "error_data": {"emoji": "invalid"}, + "errors": [], + "reason": "malformed_request", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidEmojiError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "malformed_request" in str(exc.value) + assert endpoint.called + + +async def test__add_sticker__invalid_image_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE) + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/stickers", + headers={"Authorization": "Bearer token"}, + json={"emoji": "🤔", "image": f"data:image/png;base64,{PNG_IMAGE_B64}"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.BAD_REQUEST, + json={ + "error_data": {"image": "invalid"}, + "errors": [], + "reason": "malformed_request", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(InvalidImageError) as exc: + await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert "malformed_request" in str(exc.value) + assert endpoint.called + + +async def test__add_sticker__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + await async_buffer.write(PNG_IMAGE) + await async_buffer.seek(0) + + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/stickers", + headers={"Authorization": "Bearer token"}, + json={"emoji": "🤔", "image": f"data:image/png;base64,{PNG_IMAGE_B64}"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "75bb24c9-7c08-5db0-ae3e-085929e80c54", + "emoji": "🤔", + "link": "http://cts-domain/uploads/sticker_pack/26080153-a57d-5a8c-af0e-fdecee3c4435/b4577728162f4d9ea2b35f25f9f0dcde.png?v=1626137130775", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker = await bot.add_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + emoji="🤔", + async_buffer=async_buffer, + ) + + # - Assert - + assert sticker == Sticker( + id=UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + emoji="🤔", + image_link="http://cts-domain/uploads/sticker_pack/26080153-a57d-5a8c-af0e-fdecee3c4435/b4577728162f4d9ea2b35f25f9f0dcde.png?v=1626137130775", + ) + + assert endpoint.called diff --git a/tests/client/stickers_api/test_create_sticker_pack.py b/tests/client/stickers_api/test_create_sticker_pack.py new file mode 100644 index 00000000..b6a72d37 --- /dev/null +++ b/tests/client/stickers_api/test_create_sticker_pack.py @@ -0,0 +1,68 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPack, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__create_sticker_pack__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v3/botx/stickers/packs", + headers={"Authorization": "Bearer token"}, + json={"name": "Sticker Pack"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "26080153-a57d-5a8c-af0e-fdecee3c4435", + "name": "Sticker Pack", + "public": False, + "preview": None, + "stickers": [], + "stickers_order": [], + "inserted_at": "2021-07-10T00:27:55.616703Z", + "updated_at": "2021-07-10T00:27:55.616703Z", + "deleted_at": None, + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack = await bot.create_sticker_pack(bot_id=bot_id, name="Sticker Pack") + + # - Assert - + assert sticker_pack == StickerPack( + id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + name="Sticker Pack", + is_public=False, + stickers=[], + ) + + assert endpoint.called diff --git a/tests/client/stickers_api/test_delete_sticker.py b/tests/client/stickers_api/test_delete_sticker.py new file mode 100644 index 00000000..be3c7f13 --- /dev/null +++ b/tests/client/stickers_api/test_delete_sticker.py @@ -0,0 +1,99 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__delete_sticker__sticker_or_pack_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.delete( + f"https://{host}/api/v3/botx/stickers/" + f"packs/78f9743c-8b24-4e97-8059-70908604a252/" + f"stickers/6ead1e00-f788-4ce6-9e1a-95abe219414e", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "error_data": { + "pack_id": "78f9743c-8b24-4e97-8059-70908604a252", + "sticker_id": "6ead1e00-f788-4ce6-9e1a-95abe219414e", + }, + "errors": ["Sticker or sticker pack not found."], + "reason": "not_found", + "status": "error", + }, + ), + ) + + build_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(build_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.delete_sticker( + bot_id=bot_id, + sticker_id=UUID("6ead1e00-f788-4ce6-9e1a-95abe219414e"), + sticker_pack_id=UUID("78f9743c-8b24-4e97-8059-70908604a252"), + ) + + # - Assert - + assert "Sticker or sticker pack not found" in str(exc.value) + assert endpoint.called + + +async def test__delete_sticker__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.delete( + f"https://{host}/api/v3/botx/stickers/" + f"packs/78f9743c-8b24-4e97-8059-70908604a252/" + f"stickers/6ead1e00-f788-4ce6-9e1a-95abe219414e", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "delete_sticker_from_pack_pushed", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.delete_sticker( + bot_id=bot_id, + sticker_id=UUID("6ead1e00-f788-4ce6-9e1a-95abe219414e"), + sticker_pack_id=UUID("78f9743c-8b24-4e97-8059-70908604a252"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/stickers_api/test_delete_sticker_pack.py b/tests/client/stickers_api/test_delete_sticker_pack.py new file mode 100644 index 00000000..998d6f18 --- /dev/null +++ b/tests/client/stickers_api/test_delete_sticker_pack.py @@ -0,0 +1,92 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__delete_sticker_pack__sticker_pack_not_found( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.delete( + f"https://{host}/api/v3/botx/stickers/" + f"packs/26080153-a57d-5a8c-af0e-fdecee3c4435", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "error_data": {"pack_id": "26080153-a57d-5a8c-af0e-fdecee3c4435"}, + "errors": ["Sticker pack not found."], + "reason": "pack_not_found", + "status": "error", + }, + ), + ) + + build_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(build_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.delete_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + ) + + # - Assert - + assert "pack_not_found" in str(exc) + assert endpoint.called + + +async def test__delete_sticker_pack__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.delete( + f"https://{host}/api/v3/botx/stickers/" + f"packs/26080153-a57d-5a8c-af0e-fdecee3c4435", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": "delete_sticker_pack_pushed", + }, + ), + ) + + build_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(build_bot) as bot: + await bot.delete_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + ) + + # - Assert - + assert endpoint.called diff --git a/tests/client/stickers_api/test_edit_sticker_pack.py b/tests/client/stickers_api/test_edit_sticker_pack.py new file mode 100644 index 00000000..67ffb807 --- /dev/null +++ b/tests/client/stickers_api/test_edit_sticker_pack.py @@ -0,0 +1,159 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) +from botx.models.stickers import Sticker, StickerPack + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__edit_sticker_pack__sticker_pack_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.put( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "error_data": {"pack_id": "26080153-a57d-5a8c-af0e-fdecee3c4435"}, + "errors": ["Sticker pack not found."], + "reason": "pack_not_found", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.edit_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + name="Sticker Pack 2.0", + preview=UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + stickers_order=[ + UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + ], + ) + + # - Assert - + assert "pack_not_found" in str(exc.value) + assert endpoint.called + + +async def test__edit_sticker__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.put( + f"https://{host}/api/v3/botx/stickers/packs/d881f83a-db30-4cff-b60e-f24ac53deecf", + headers={"Authorization": "Bearer token"}, + json={ + "name": "Sticker Pack 2.0", + "preview": "528c3953-5842-5a30-b2cb-8a09218497bc", + "stickers_order": [ + "75bb24c9-7c08-5db0-ae3e-085929e80c54", + "528c3953-5842-5a30-b2cb-8a09218497bc", + ], + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "d881f83a-db30-4cff-b60e-f24ac53deecf", + "name": "Sticker Pack 2.0", + "public": True, + "preview": "528c3953-5842-5a30-b2cb-8a09218497bc", + "stickers_order": [ + "75bb24c9-7c08-5db0-ae3e-085929e80c54", + "528c3953-5842-5a30-b2cb-8a09218497bc", + ], + "stickers": [ + { + "id": "75bb24c9-7c08-5db0-ae3e-085929e80c54", + "emoji": "🤔", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + { + "id": "528c3953-5842-5a30-b2cb-8a09218497bc", + "emoji": "😀", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + ], + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2021-07-22T13:26:41.562143Z", + "deleted_at": None, + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack = await bot.edit_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + name="Sticker Pack 2.0", + preview=UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + stickers_order=[ + UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + ], + ) + + # - Assert - + assert sticker_pack == StickerPack( + id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + name="Sticker Pack 2.0", + is_public=True, + stickers=[ + Sticker( + id=UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + emoji="🤔", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ), + Sticker( + id=UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + emoji="😀", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ), + ], + ) + + assert endpoint.called diff --git a/tests/client/stickers_api/test_get_sticker.py b/tests/client/stickers_api/test_get_sticker.py new file mode 100644 index 00000000..5092f35e --- /dev/null +++ b/tests/client/stickers_api/test_get_sticker.py @@ -0,0 +1,109 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) +from botx.models.stickers import Sticker + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__get_sticker__sticker_pack_or_sticker_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/" + f"stickers/75bb24c9-7c08-5db0-ae3e-085929e80c54", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "error_data": { + "pack_id": "26080153-a57d-5a8c-af0e-fdecee3c4435", + "sticker_id": "75bb24c9-7c08-5db0-ae3e-085929e80c54", + }, + "errors": ["Sticker or sticker pack not found."], + "reason": "not_found", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.get_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + sticker_id=UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + ) + + # - Assert - + assert "not_found" in str(exc.value) + assert endpoint.called + + +async def test__get_sticker__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435/" + f"stickers/75bb24c9-7c08-5db0-ae3e-085929e80c54", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "75bb24c9-7c08-5db0-ae3e-085929e80c54", + "emoji": "🤔", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "preview": "https://cts-host/uploads/sticker_pack/image.png", + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker = await bot.get_sticker( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + sticker_id=UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + ) + + # - Assert - + assert sticker == Sticker( + id=UUID("75bb24c9-7c08-5db0-ae3e-085929e80c54"), + emoji="🤔", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ) + + assert endpoint.called diff --git a/tests/client/stickers_api/test_get_sticker_pack.py b/tests/client/stickers_api/test_get_sticker_pack.py new file mode 100644 index 00000000..d344b0c4 --- /dev/null +++ b/tests/client/stickers_api/test_get_sticker_pack.py @@ -0,0 +1,202 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + StickerPackOrStickerNotFoundError, + lifespan_wrapper, +) +from botx.models.stickers import Sticker, StickerPack + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__get_sticker__sticker_pack_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs/26080153-a57d-5a8c-af0e-fdecee3c4435", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "error_data": {"pack_id": "26080153-a57d-5a8c-af0e-fdecee3c4435"}, + "errors": ["Sticker pack not found."], + "reason": "pack_not_found", + "status": "error", + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(StickerPackOrStickerNotFoundError) as exc: + await bot.get_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + ) + + # - Assert - + assert "pack_not_found" in str(exc.value) + assert endpoint.called + + +async def test__get_sticker_pack__stickers_in_right_order_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs/d881f83a-db30-4cff-b60e-f24ac53deecf", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "d881f83a-db30-4cff-b60e-f24ac53deecf", + "name": "Sticker Pack", + "public": True, + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "stickers_order": [ + "528c3953-5842-5a30-b2cb-8a09218497bc", + "750bb400-bb37-4ff9-aa92-cc293f09cafa", + ], + "stickers": [ + { + "id": "750bb400-bb37-4ff9-aa92-cc293f09cafa", + "emoji": "🤔", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + { + "id": "528c3953-5842-5a30-b2cb-8a09218497bc", + "emoji": "🤔", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + ], + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack = await bot.get_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + ) + + # - Assert - + assert sticker_pack == StickerPack( + id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + name="Sticker Pack", + is_public=True, + stickers=[ + Sticker( + id=UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + emoji="🤔", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ), + Sticker( + id=UUID("750bb400-bb37-4ff9-aa92-cc293f09cafa"), + emoji="🤔", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ), + ], + ) + + assert endpoint.called + + +async def test__get_sticker_pack__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs/d881f83a-db30-4cff-b60e-f24ac53deecf", + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "id": "d881f83a-db30-4cff-b60e-f24ac53deecf", + "name": "Sticker Pack", + "public": True, + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "stickers_order": [], + "stickers": [ + { + "id": "528c3953-5842-5a30-b2cb-8a09218497bc", + "emoji": "🤔", + "link": "https://cts-host/uploads/sticker_pack/image.png", + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + ], + "inserted_at": "2020-12-28T12:56:43.672163Z", + "updated_at": "2020-12-28T12:56:43.672163Z", + "deleted_at": None, + }, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack = await bot.get_sticker_pack( + bot_id=bot_id, + sticker_pack_id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + ) + + # - Assert - + assert sticker_pack == StickerPack( + id=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + name="Sticker Pack", + is_public=True, + stickers=[ + Sticker( + id=UUID("528c3953-5842-5a30-b2cb-8a09218497bc"), + emoji="🤔", + image_link="https://cts-host/uploads/sticker_pack/image.png", + ), + ], + ) + + assert endpoint.called diff --git a/tests/client/stickers_api/test_get_sticker_packs.py b/tests/client/stickers_api/test_get_sticker_packs.py new file mode 100644 index 00000000..297650c0 --- /dev/null +++ b/tests/client/stickers_api/test_get_sticker_packs.py @@ -0,0 +1,228 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, lifespan_wrapper +from botx.models.stickers import StickerPackFromList + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__iterate_by_sticker_packs__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs", + headers={"Authorization": "Bearer token"}, + params={"user_huid": "d881f83a-db30-4cff-b60e-f24ac53deecf"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "packs": [ + { + "id": "26080153-a57d-5a8c-af0e-fdecee3c4435", + "name": "Sticker Pack", + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "public": True, + "stickers_count": 2, + "stickers_order": [ + "a998f599-d7ac-5e04-9fdb-2d98224ce4ff", + "25054ac4-8be2-5a4b-ae00-9efd38c73fb7", + ], + "inserted_at": "2020-11-28T12:56:43.672163Z", + "updated_at": "2021-02-18T12:52:31.571133Z", + "deleted_at": None, + }, + ], + "pagination": { + "after": None, + }, + }, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + sticker_pack_list = [] + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack_pages = bot.iterate_by_sticker_packs( + bot_id=bot_id, + user_huid=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + ) + async for sticker_packs in sticker_pack_pages: + sticker_pack_list.append(sticker_packs) + + # - Assert - + assert sticker_pack_list == [ + StickerPackFromList( + id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + name="Sticker Pack", + is_public=True, + stickers_count=2, + sticker_ids=[ + UUID("a998f599-d7ac-5e04-9fdb-2d98224ce4ff"), + UUID("25054ac4-8be2-5a4b-ae00-9efd38c73fb7"), + ], + ), + ] + assert endpoint.called + + +async def test__iterate_by_sticker_packs__iterate_by_pages_succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + monkeypatch: pytest.MonkeyPatch, +) -> None: + # - Arrange - + monkeypatch.setattr("botx.bot.bot.STICKER_PACKS_PER_PAGE", 2) + + # Mock order matters + # https://lundberg.github.io/respx/guide/#routing-requests + second_sticker_endpoint_call = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs", + headers={"Authorization": "Bearer token"}, + params={ + "user_huid": "d881f83a-db30-4cff-b60e-f24ac53deecf", + "limit": 2, + "after": "base64string", + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "packs": [ + { + "id": "750bb400-bb37-4ff9-aa92-cc293f09cafa", + "name": "Sticker Pack 3", + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "public": True, + "stickers_count": 2, + "stickers_order": [ + "a998f599-d7ac-5e04-9fdb-2d98224ce4ff", + "25054ac4-8be2-5a4b-ae00-9efd38c73fb7", + ], + "inserted_at": "2020-11-28T12:56:43.672163Z", + "updated_at": "2021-02-18T12:52:31.571133Z", + "deleted_at": None, + }, + ], + "pagination": { + "after": None, + }, + }, + }, + ), + ) + first_sticker_endpoint_call = respx_mock.get( + f"https://{host}/api/v3/botx/stickers/packs", + headers={"Authorization": "Bearer token"}, + params={"user_huid": "d881f83a-db30-4cff-b60e-f24ac53deecf", "limit": 2}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "packs": [ + { + "id": "26080153-a57d-5a8c-af0e-fdecee3c4435", + "name": "Sticker Pack 1", + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "public": True, + "stickers_count": 2, + "stickers_order": [ + "a998f599-d7ac-5e04-9fdb-2d98224ce4ff", + "25054ac4-8be2-5a4b-ae00-9efd38c73fb7", + ], + "inserted_at": "2020-11-28T12:56:43.672163Z", + "updated_at": "2021-02-18T12:52:31.571133Z", + "deleted_at": None, + }, + { + "id": "89152263-2484-4e00-bc6c-90003027e39e", + "name": "Sticker Pack 2", + "preview": "https://cts-host/uploads/sticker_pack/image.png", + "public": True, + "stickers_count": 1, + "stickers_order": [ + "a998f599-d7ac-5e04-9fdb-2d98224ce4ff", + ], + "inserted_at": "2020-11-28T12:56:43.672163Z", + "updated_at": "2021-02-18T12:52:31.571133Z", + "deleted_at": None, + }, + ], + "pagination": { + "after": "base64string", + }, + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + sticker_pack_list = [] + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + sticker_pack_pages = bot.iterate_by_sticker_packs( + bot_id=bot_id, + user_huid=UUID("d881f83a-db30-4cff-b60e-f24ac53deecf"), + ) + async for sticker_packs in sticker_pack_pages: + sticker_pack_list.append(sticker_packs) + + # - Assert - + assert sticker_pack_list == [ + StickerPackFromList( + id=UUID("26080153-a57d-5a8c-af0e-fdecee3c4435"), + name="Sticker Pack 1", + is_public=True, + stickers_count=2, + sticker_ids=[ + UUID("a998f599-d7ac-5e04-9fdb-2d98224ce4ff"), + UUID("25054ac4-8be2-5a4b-ae00-9efd38c73fb7"), + ], + ), + StickerPackFromList( + id=UUID("89152263-2484-4e00-bc6c-90003027e39e"), + name="Sticker Pack 2", + is_public=True, + stickers_count=1, + sticker_ids=[ + UUID("a998f599-d7ac-5e04-9fdb-2d98224ce4ff"), + ], + ), + StickerPackFromList( + id=UUID("750bb400-bb37-4ff9-aa92-cc293f09cafa"), + name="Sticker Pack 3", + is_public=True, + stickers_count=2, + sticker_ids=[ + UUID("a998f599-d7ac-5e04-9fdb-2d98224ce4ff"), + UUID("25054ac4-8be2-5a4b-ae00-9efd38c73fb7"), + ], + ), + ] + assert first_sticker_endpoint_call.called + assert second_sticker_endpoint_call.called diff --git a/tests/client/test_authorized_botx_method.py b/tests/client/test_authorized_botx_method.py new file mode 100644 index 00000000..9da35c8a --- /dev/null +++ b/tests/client/test_authorized_botx_method.py @@ -0,0 +1,176 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import BotAccountWithSecret, InvalidBotAccountError +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.client.authorized_botx_method import AuthorizedBotXMethod +from tests.client.test_botx_method import ( + BotXAPIFooBarRequestPayload, + BotXAPIFooBarResponsePayload, +) + + +class FooBarMethod(AuthorizedBotXMethod): + async def execute( + self, + payload: BotXAPIFooBarRequestPayload, + ) -> BotXAPIFooBarResponsePayload: + path = "/foo/bar" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPIFooBarResponsePayload, + response, + ) + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__authorized_botx_method__unauthorized( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_signature: str, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + token_endpoint = respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": "token", + }, + ), + ) + + foo_bar_endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response(HTTPStatus.UNAUTHORIZED), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(InvalidBotAccountError) as exc: + await method.execute(payload) + + # - Assert - + assert "failed with code 401" in str(exc.value) + assert token_endpoint.called + assert foo_bar_endpoint.called + + +async def test__authorized_botx_method__succeed( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_signature: str, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + token_endpoint = respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": "token", + }, + ), + ) + + foo_bar_endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + botx_api_foo_bar = await method.execute(payload) + + # - Assert - + assert botx_api_foo_bar.to_domain() == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert token_endpoint.called + assert foo_bar_endpoint.called + + +async def test__authorized_botx_method__with_prepared_token( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + prepared_bot_accounts_storage: BotAccountsStorage, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + prepared_bot_accounts_storage, + ) + + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + botx_api_foo_bar = await method.execute(payload) + + # - Assert - + assert botx_api_foo_bar.to_domain() == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called diff --git a/tests/client/test_botx_method.py b/tests/client/test_botx_method.py new file mode 100644 index 00000000..f87a48aa --- /dev/null +++ b/tests/client/test_botx_method.py @@ -0,0 +1,242 @@ +from http import HTTPStatus +from typing import Literal +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + BotAccountWithSecret, + InvalidBotXResponsePayloadError, + InvalidBotXStatusCodeError, +) +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.client.botx_method import BotXMethod, response_exception_thrower +from botx.client.exceptions.base import BaseClientError +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class FooBarError(BaseClientError): + """Test exception.""" + + +class BotXAPIFooBarRequestPayload(UnverifiedPayloadBaseModel): + baz: int + + @classmethod + def from_domain(cls, baz: int) -> "BotXAPIFooBarRequestPayload": + return cls(baz=baz) + + +class BotXAPISyncIdResult(VerifiedPayloadBaseModel): + sync_id: UUID + + +class BotXAPIFooBarResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + result: BotXAPISyncIdResult + + def to_domain(self) -> UUID: + return self.result.sync_id + + +class FooBarMethod(BotXMethod): + status_handlers = { + 403: response_exception_thrower(FooBarError, "FooBar comment"), + } + + async def execute( + self, + payload: BotXAPIFooBarRequestPayload, + ) -> BotXAPIFooBarResponsePayload: + path = "/foo/bar" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + return self._verify_and_extract_api_model( + BotXAPIFooBarResponsePayload, + response, + ) + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__botx_method__invalid_botx_status_code_error_raised( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response(HTTPStatus.METHOD_NOT_ALLOWED), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(InvalidBotXStatusCodeError) as exc: + await method.execute(payload) + + # - Assert - + assert "failed with code 405" in str(exc.value) + assert endpoint.called + + +async def test__botx_method__invalid_json_raises_invalid_botx_response_payload_error( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + content='{"invalid": "json', + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(InvalidBotXResponsePayloadError) as exc: + await method.execute(payload) + + # - Assert - + assert '{"invalid": "json' in str(exc.value) + assert endpoint.called + + +async def test__botx_method__invalid_schema_raises_invalid_botx_response_payload_error( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"invalid": "schema"}, + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(InvalidBotXResponsePayloadError) as exc: + await method.execute(payload) + + # - Assert - + assert '{"invalid": "schema"}' in str(exc.value) + assert endpoint.called + + +async def test__botx_method__status_handler_called( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response(HTTPStatus.FORBIDDEN), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(FooBarError) as exc: + await method.execute(payload) + + # - Assert - + assert "403" in str(exc.value) + assert "FooBar comment" in str(exc.value) + assert endpoint.called + + +async def test__botx_method__succeed( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + botx_api_foo_bar = await method.execute(payload) + + # - Assert - + assert botx_api_foo_bar.to_domain() == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called diff --git a/tests/client/test_botx_method_callback.py b/tests/client/test_botx_method_callback.py new file mode 100644 index 00000000..7cee4c52 --- /dev/null +++ b/tests/client/test_botx_method_callback.py @@ -0,0 +1,500 @@ +# type: ignore [attr-defined] + +import asyncio +import types +from http import HTTPStatus +from typing import Optional +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + BotShuttingDownError, + BotXMethodCallbackNotFoundError, + BotXMethodFailedCallbackReceivedError, + CallbackNotReceivedError, + HandlerCollector, + lifespan_wrapper, +) +from botx.client.botx_method import ( + BotXMethod, + ErrorCallbackHandlers, + callback_exception_thrower, +) +from botx.client.exceptions.base import BaseClientError +from botx.missing import MissingOptional, Undefined, not_undefined +from botx.models.method_callbacks import BotAPIMethodSuccessfulCallback +from tests.client.test_botx_method import ( + BotXAPIFooBarRequestPayload, + BotXAPIFooBarResponsePayload, +) + + +class FooBarError(BaseClientError): + """Test exception.""" + + +class FooBarCallbackMethod(BotXMethod): + error_callback_handlers: ErrorCallbackHandlers = { + "foo_bar_error": callback_exception_thrower( + FooBarError, + "FooBar comment", + ), + } + + async def execute( + self, + payload: BotXAPIFooBarRequestPayload, + wait_callback: bool, + callback_timeout: MissingOptional[int] = Undefined, + ) -> BotXAPIFooBarResponsePayload: + path = "/foo/bar" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + api_model = self._verify_and_extract_api_model( + BotXAPIFooBarResponsePayload, + response, + ) + + await self._process_callback( + api_model.result.sync_id, + wait_callback, + callback_timeout, + ) + + return api_model + + +async def call_foo_bar( + self: Bot, + bot_id: UUID, + baz: int, + wait_callback: bool = True, + callback_timeout: Optional[int] = None, +) -> UUID: + method = FooBarCallbackMethod( + bot_id, + self._httpx_client, + self._bot_accounts_storage, + self._callback_manager, + ) + + payload = BotXAPIFooBarRequestPayload.from_domain(baz=baz) + botx_api_foo_bar = await method.execute( + payload, + wait_callback, + not_undefined(callback_timeout, self.default_callback_timeout), + ) + + return botx_api_foo_bar.to_domain() + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__botx_method_callback__callback_not_found( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(BotXMethodCallbackNotFoundError) as exc: + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "chat_not_found", + "errors": [], + "error_data": { + "group_chat_id": "705df263-6bfd-536a-9d51-13524afaab5c", + "error_description": ( + "Chat with id 705df263-6bfd-536a-9d51-13524afaab5c not found" + ), + }, + }, + ) + + # - Assert - + assert "No callback found" in str(exc.value) + assert "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3" in str(exc.value) + + +async def test__botx_method_callback__error_callback_error_handler_called( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.call_foo_bar(bot_id, baz=1), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "foo_bar_error", + "errors": [], + "error_data": { + "group_chat_id": "705df263-6bfd-536a-9d51-13524afaab5c", + "error_description": ( + "Chat with id 705df263-6bfd-536a-9d51-13524afaab5c not found" + ), + }, + }, + ) + + with pytest.raises(FooBarError) as exc: + await task + + # - Assert - + assert "foo_bar_error" in str(exc.value) + assert "FooBar comment" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_callback__error_callback_received( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.call_foo_bar(bot_id, baz=1), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "quux_error", + "errors": [], + "error_data": { + "group_chat_id": "705df263-6bfd-536a-9d51-13524afaab5c", + "error_description": ( + "Chat with id 705df263-6bfd-536a-9d51-13524afaab5c not found" + ), + }, + }, + ) + + with pytest.raises(BotXMethodFailedCallbackReceivedError) as exc: + await task + + # - Assert - + assert "failed with" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_callback__cancelled_callback_future_during_shutdown( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(CallbackNotReceivedError) as exc: + await bot.call_foo_bar(bot_id, baz=1, callback_timeout=0) + + # - Assert - + assert "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_callback__callback_received_after_timeout( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(CallbackNotReceivedError) as exc: + await bot.call_foo_bar(bot_id, baz=1, callback_timeout=0) + + bot.set_raw_botx_method_result( + { + "status": "error", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "reason": "quux_error", + "errors": [], + "error_data": { + "group_chat_id": "705df263-6bfd-536a-9d51-13524afaab5c", + "error_description": ( + "Chat with id 705df263-6bfd-536a-9d51-13524afaab5c not found" + ), + }, + }, + ) + + # - Assert - + assert "hasn't been received" in str(exc.value) + assert "don't wait callback" in loguru_caplog.text + assert "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3" in loguru_caplog.text + assert endpoint.called + + +async def test__botx_method_callback__dont_wait_for_callback( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + foo_bar = await bot.call_foo_bar(bot_id, baz=1, wait_callback=False) + + # - Assert - + assert foo_bar == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__botx_method_callback__pending_callback_future_during_shutdown( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.call_foo_bar(bot_id, baz=1), + ) + await asyncio.sleep(0) # Return control to event loop + + with pytest.raises(BotShuttingDownError) as exc: + await task + + # - Assert - + assert "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_callback__callback_successful_received( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + task = asyncio.create_task( + bot.call_foo_bar(bot_id, baz=1), + ) + await asyncio.sleep(0) # Return control to event loop + + bot.set_raw_botx_method_result( + { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + }, + ) + + # - Assert - + assert await task == UUID("21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3") + assert endpoint.called + + +async def test__botx_method_callback__bot_wait_callback( + respx_mock: MockRouter, + httpx_client: httpx.AsyncClient, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={"baz": 1}, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + built_bot = Bot( + collectors=[HandlerCollector()], + bot_accounts=[bot_account], + httpx_client=httpx_client, + ) + + built_bot.call_foo_bar = types.MethodType(call_foo_bar, built_bot) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + foo_bar = await bot.call_foo_bar(bot_id, baz=1, wait_callback=False) + task = asyncio.create_task(bot.wait_botx_method_callback(foo_bar, None)) + + # Return control to event loop + await asyncio.sleep(0) + + bot.set_raw_botx_method_result( + { + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "status": "ok", + "result": {}, + }, + ) + + callback = await task + + # - Assert - + assert callback == BotAPIMethodSuccessfulCallback( + sync_id=foo_bar, + status="ok", + result={}, + ) + assert endpoint.called diff --git a/tests/client/test_botx_method_stream.py b/tests/client/test_botx_method_stream.py new file mode 100644 index 00000000..829939a2 --- /dev/null +++ b/tests/client/test_botx_method_stream.py @@ -0,0 +1,137 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import BotAccountWithSecret, InvalidBotXStatusCodeError +from botx.async_buffer import AsyncBufferWritable +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.client.botx_method import BotXMethod, response_exception_thrower +from botx.client.exceptions.base import BaseClientError +from tests.client.test_botx_method import BotXAPIFooBarRequestPayload + + +class FooBarError(BaseClientError): + """Test exception.""" + + +class FooBarStreamMethod(BotXMethod): + status_handlers = { + 403: response_exception_thrower(FooBarError), + } + + async def execute( + self, + payload: BotXAPIFooBarRequestPayload, + async_buffer: AsyncBufferWritable, + ) -> None: + path = "/foo/bar" + + async with self._botx_method_stream( + "GET", + self._build_url(path), + params=payload.jsonable_dict(), + ) as response: + async for chunk in response.aiter_bytes(): + await async_buffer.write(chunk) + + await async_buffer.seek(0) + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__botx_method_stream__invalid_botx_status_code_error_raised( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get(f"https://{host}/foo/bar", params={"baz": 1}).mock( + return_value=httpx.Response(HTTPStatus.METHOD_NOT_ALLOWED), + ) + + method = FooBarStreamMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(InvalidBotXStatusCodeError) as exc: + await method.execute(payload, async_buffer) + + # - Assert - + assert "failed with code 405" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_stream__status_handler_called( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get(f"https://{host}/foo/bar", params={"baz": 1}).mock( + return_value=httpx.Response(HTTPStatus.FORBIDDEN), + ) + + method = FooBarStreamMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + with pytest.raises(FooBarError) as exc: + await method.execute(payload, async_buffer) + + # - Assert - + assert "403" in str(exc.value) + assert endpoint.called + + +async def test__botx_method_stream__succeed( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, +) -> None: + # - Arrange - + endpoint = respx_mock.get(f"https://{host}/foo/bar", params={"baz": 1}).mock( + return_value=httpx.Response( + HTTPStatus.OK, + content=b"Hello, world!\n", + ), + ) + + method = FooBarStreamMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain(baz=1) + + # - Act - + await method.execute(payload, async_buffer) + + # - Assert - + assert await async_buffer.read() == b"Hello, world!\n" + assert endpoint.called diff --git a/tests/client/test_botx_method_undefined_cleaned.py b/tests/client/test_botx_method_undefined_cleaned.py new file mode 100644 index 00000000..ed4b418c --- /dev/null +++ b/tests/client/test_botx_method_undefined_cleaned.py @@ -0,0 +1,113 @@ +from http import HTTPStatus +from typing import Any, Dict, Literal +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import BotAccountWithSecret +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.client.botx_method import BotXMethod +from botx.missing import Undefined +from botx.models.api_base import UnverifiedPayloadBaseModel, VerifiedPayloadBaseModel + + +class BotXAPIFooBarRequestPayload(UnverifiedPayloadBaseModel): + baz: Dict[str, Any] + + @classmethod + def from_domain(cls, baz: Dict[str, Any]) -> "BotXAPIFooBarRequestPayload": + return cls(baz=baz) + + +class BotXAPIFooBarResponsePayload(VerifiedPayloadBaseModel): + status: Literal["ok"] + + +class FooBarMethod(BotXMethod): + async def execute( + self, + payload: BotXAPIFooBarRequestPayload, + ) -> None: + path = "/foo/bar" + + response = await self._botx_method_call( + "POST", + self._build_url(path), + json=payload.jsonable_dict(), + ) + + self._verify_and_extract_api_model( + BotXAPIFooBarResponsePayload, + response, + ) + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__botx_method__undefined_cleaned( + httpx_client: httpx.AsyncClient, + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/foo/bar", + json={ + "baz": { + "key": [ + { + "key1": "value", + "key3": [1, 2, 3], + "key4": {"key1": "value"}, + "key7": {}, + "key8": [], + }, + ], + }, + }, + headers={"Content-Type": "application/json"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={"status": "ok"}, + ), + ) + + method = FooBarMethod( + bot_id, + httpx_client, + BotAccountsStorage([bot_account]), + ) + payload = BotXAPIFooBarRequestPayload.from_domain( + baz={ + "key": [ + { + "key1": "value", + "key2": Undefined, + "key3": [Undefined, 1, 2, Undefined, 3], + "key4": {"key1": "value", "key2": Undefined}, + "key5": [Undefined, Undefined], + "key6": {"key1": Undefined, "key2": Undefined}, + "key7": {}, + "key8": [], + }, + { + "key": Undefined, + }, + ], + }, + ) + + # - Act - + await method.execute(payload) + + # - Assert - + assert endpoint.called diff --git a/tests/test_clients/test_methods/__init__.py b/tests/client/users_api/__init__.py similarity index 100% rename from tests/test_clients/test_methods/__init__.py rename to tests/client/users_api/__init__.py diff --git a/tests/client/users_api/test_search_user_by_email.py b/tests/client/users_api/test_search_user_by_email.py new file mode 100644 index 00000000..ccaf71ba --- /dev/null +++ b/tests/client/users_api/test_search_user_by_email.py @@ -0,0 +1,113 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + UserFromSearch, + UserNotFoundError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__search_user_by_email__user_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_email", + headers={"Authorization": "Bearer token"}, + params={"email": "ad_user@cts.com"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "user_not_found", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UserNotFoundError) as exc: + await bot.search_user_by_email( + bot_id=bot_id, + email="ad_user@cts.com", + ) + + # - Assert - + assert "user_not_found" in str(exc.value) + assert endpoint.called + + +async def test__search_user_by_email__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_email", + headers={"Authorization": "Bearer token"}, + params={"email": "ad_user@cts.com"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "user_huid": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "ad_login": "ad_user_login", + "ad_domain": "cts.com", + "name": "Bob", + "company": "Bobs Co", + "company_position": "Director", + "department": "Owners", + "emails": ["ad_user@cts.com"], + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + user = await bot.search_user_by_email( + bot_id=bot_id, + email="ad_user@cts.com", + ) + + # - Assert - + assert user == UserFromSearch( + huid=UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + ad_login="ad_user_login", + ad_domain="cts.com", + username="Bob", + company="Bobs Co", + company_position="Director", + department="Owners", + emails=["ad_user@cts.com"], + ) + + assert endpoint.called diff --git a/tests/client/users_api/test_search_user_by_huid.py b/tests/client/users_api/test_search_user_by_huid.py new file mode 100644 index 00000000..aebcca8f --- /dev/null +++ b/tests/client/users_api/test_search_user_by_huid.py @@ -0,0 +1,113 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + UserFromSearch, + UserNotFoundError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__search_user_by_huid__user_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_huid", + headers={"Authorization": "Bearer token"}, + params={"user_huid": "f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "user_not_found", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UserNotFoundError) as exc: + await bot.search_user_by_huid( + bot_id=bot_id, + huid=UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"), + ) + + # - Assert - + assert "user_not_found" in str(exc.value) + assert endpoint.called + + +async def test__search_user_by_huid__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_huid", + headers={"Authorization": "Bearer token"}, + params={"user_huid": "f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "user_huid": "f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1", + "ad_login": "ad_user_login", + "ad_domain": "cts.com", + "name": "Bob", + "company": "Bobs Co", + "company_position": "Director", + "department": "Owners", + "emails": ["ad_user@cts.com"], + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + user = await bot.search_user_by_huid( + bot_id=bot_id, + huid=UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"), + ) + + # - Assert - + assert user == UserFromSearch( + huid=UUID("f837dff4-d3ad-4b8d-a0a3-5c6ca9c747d1"), + ad_login="ad_user_login", + ad_domain="cts.com", + username="Bob", + company="Bobs Co", + company_position="Director", + department="Owners", + emails=["ad_user@cts.com"], + ) + + assert endpoint.called diff --git a/tests/client/users_api/test_search_user_by_login.py b/tests/client/users_api/test_search_user_by_login.py new file mode 100644 index 00000000..3a2ec0e2 --- /dev/null +++ b/tests/client/users_api/test_search_user_by_login.py @@ -0,0 +1,115 @@ +from http import HTTPStatus +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + UserFromSearch, + UserNotFoundError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__search_user_by_ad__user_not_found_error_raised( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_login", + headers={"Authorization": "Bearer token"}, + params={"ad_login": "ad_user_login", "ad_domain": "cts.com"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.NOT_FOUND, + json={ + "status": "error", + "reason": "user_not_found", + "errors": [], + "error_data": {}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UserNotFoundError) as exc: + await bot.search_user_by_ad( + bot_id=bot_id, + ad_login="ad_user_login", + ad_domain="cts.com", + ) + + # - Assert - + assert "user_not_found" in str(exc.value) + assert endpoint.called + + +async def test__search_user_by_ad__succeed( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/users/by_login", + headers={"Authorization": "Bearer token"}, + params={"ad_login": "ad_user_login", "ad_domain": "cts.com"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": { + "user_huid": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "ad_login": "ad_user_login", + "ad_domain": "cts.com", + "name": "Bob", + "company": "Bobs Co", + "company_position": "Director", + "department": "Owners", + "emails": ["ad_user@cts.com"], + }, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + user = await bot.search_user_by_ad( + bot_id=bot_id, + ad_login="ad_user_login", + ad_domain="cts.com", + ) + + # - Assert - + assert user == UserFromSearch( + huid=UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + ad_login="ad_user_login", + ad_domain="cts.com", + username="Bob", + company="Bobs Co", + company_position="Director", + department="Owners", + emails=["ad_user@cts.com"], + ) + + assert endpoint.called diff --git a/tests/conftest.py b/tests/conftest.py index 71bbddc9..5ec584e1 100644 --- a/tests/conftest.py +++ b/tests/conftest.py @@ -1,10 +1,252 @@ -pytest_plugins = ( - "tests.fixtures.bot", - "tests.fixtures.collector", - "tests.fixtures.handlers", - "tests.fixtures.storage", - "tests.fixtures.credentials", - "tests.fixtures.logging", - "tests.fixtures.messages", - "tests.fixtures.errors", +import logging +from datetime import datetime +from http import HTTPStatus +from typing import Any, AsyncGenerator, Callable, Dict, Generator, List, Optional +from unittest.mock import Mock +from uuid import UUID, uuid4 + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from pydantic import BaseModel +from respx.router import MockRouter + +from botx import ( + BotAccount, + BotAccountWithSecret, + Chat, + ChatTypes, + IncomingMessage, + UserDevice, + UserSender, ) +from botx.bot.bot_accounts_storage import BotAccountsStorage +from botx.logger import logger + + +@pytest.fixture(autouse=True) +def enable_logger() -> None: + logger.enable("botx") + + +@pytest.fixture +def prepared_bot_accounts_storage( + bot_id: UUID, + bot_account: BotAccountWithSecret, +) -> BotAccountsStorage: + bot_accounts_storage = BotAccountsStorage([bot_account]) + bot_accounts_storage.set_token(bot_id, "token") + + return bot_accounts_storage + + +@pytest.fixture +def datetime_formatter() -> Callable[[str], datetime]: + class DateTimeFormatter(BaseModel): # noqa: WPS431 + value: datetime + + def factory(dt_str: str) -> datetime: + return DateTimeFormatter(value=dt_str).value + + return factory + + +@pytest.fixture +def host() -> str: + return "cts.example.com" + + +@pytest.fixture +def bot_id() -> UUID: + return UUID("24348246-6791-4ac0-9d86-b948cd6a0e46") + + +@pytest.fixture +def bot_account(host: str, bot_id: UUID) -> BotAccountWithSecret: + return BotAccountWithSecret( + id=bot_id, + host=host, + secret_key="bee001", + ) + + +@pytest.fixture +def bot_signature() -> str: + return "E050AEEA197E0EF0A6E1653E18B7D41C7FDEC0FCFBA44C44FCCD2A88CEABD130" + + +@pytest.fixture +def mock_authorization( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_signature: str, +) -> None: + """Fixture should be used as a marker.""" + respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + json={ + "status": "ok", + "result": "token", + }, + ), + ) + + +@pytest.hookimpl(trylast=True) +def pytest_collection_modifyitems(items: List[pytest.Function]) -> None: + for item in items: + if item.get_closest_marker("mock_authorization"): + item.fixturenames.append("mock_authorization") + + +@pytest.fixture() +def loguru_caplog( + caplog: pytest.LogCaptureFixture, +) -> Generator[pytest.LogCaptureFixture, None, None]: + # https://github.com/Delgan/loguru/issues/59 + + class PropogateHandler(logging.Handler): # noqa: WPS431 + def emit(self, record: logging.LogRecord) -> None: + logging.getLogger(record.name).handle(record) + + handler_id = logger.add(PropogateHandler(), format="{message}") + yield caplog + logger.remove(handler_id) + + +@pytest.fixture +async def httpx_client() -> AsyncGenerator[httpx.AsyncClient, None]: + async with httpx.AsyncClient() as client: + yield client + + +@pytest.fixture +async def async_buffer() -> AsyncGenerator[NamedTemporaryFile, None]: + async with NamedTemporaryFile("wb+") as async_buffer: + yield async_buffer + + +@pytest.fixture +def api_incoming_message_factory() -> Callable[..., Dict[str, Any]]: + def decorator( + *, + bot_id: Optional[UUID] = None, + group_chat_id: Optional[UUID] = None, + user_huid: Optional[UUID] = None, + host: Optional[str] = None, + attachment: Optional[Dict[str, Any]] = None, + async_file: Optional[Dict[str, Any]] = None, + ) -> Dict[str, Any]: + return { + "bot_id": str(bot_id) if bot_id else "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "/hello", + "command_type": "user", + "data": {}, + "metadata": {}, + }, + "attachments": [attachment] if attachment else [], + "async_files": [async_file] if async_file else [], + "source_sync_id": None, + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": False, + "timezone": "Europe/Moscow", + }, + "device_software": None, + "group_chat_id": ( + str(group_chat_id) + if group_chat_id + else "30dc1980-643a-00ad-37fc-7cc10d74e935" + ), + "host": host or "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": ( + str(user_huid) + if user_huid + else "f16cdc5f-6366-5552-9ecd-c36290ab3d11" + ), + "username": None, + }, + "proto_version": 4, + "entities": [], + } + + return decorator + + +@pytest.fixture +def incoming_message_factory( + bot_id: UUID, +) -> Callable[..., IncomingMessage]: + def decorator( + *, + body: str = "", + ad_login: Optional[str] = None, + ad_domain: Optional[str] = None, + ) -> IncomingMessage: + return IncomingMessage( + bot=BotAccount( + id=bot_id, + host="cts.example.com", + ), + sync_id=uuid4(), + source_sync_id=None, + body=body, + data={}, + metadata={}, + sender=UserSender( + huid=uuid4(), + ad_login=ad_login, + ad_domain=ad_domain, + username=None, + is_chat_admin=True, + is_chat_creator=True, + device=UserDevice( + manufacturer=None, + device_name=None, + os=None, + pushes=None, + timezone=None, + permissions=None, + platform=None, + platform_package_id=None, + app_version=None, + locale=None, + ), + ), + chat=Chat( + id=uuid4(), + type=ChatTypes.PERSONAL_CHAT, + ), + raw_command=None, + ) + + return decorator + + +@pytest.fixture +def correct_handler_trigger() -> Mock: + return Mock() + + +@pytest.fixture +def incorrect_handler_trigger() -> Mock: + return Mock() diff --git a/tests/fixtures/bot.py b/tests/fixtures/bot.py deleted file mode 100644 index e164dd8e..00000000 --- a/tests/fixtures/bot.py +++ /dev/null @@ -1,21 +0,0 @@ -import pytest - -from botx import Bot, BotXCredentials, TestClient - - -@pytest.fixture() -def bot(host, secret_key, bot_id, token): - accounts = BotXCredentials( - host=host, - secret_key=secret_key, - bot_id=bot_id, - token=token, - ) - - return Bot(bot_accounts=[accounts]) - - -@pytest.fixture() -def client(bot): - with TestClient(bot) as client: - yield client diff --git a/tests/fixtures/collector.py b/tests/fixtures/collector.py deleted file mode 100644 index 851be7a3..00000000 --- a/tests/fixtures/collector.py +++ /dev/null @@ -1,75 +0,0 @@ -from typing import Callable, Optional - -import pytest - -from botx import Collector, Depends, Message - - -def build_botx_handler(name: Optional[str] = None) -> Callable[[], None]: - def factory(_message: Message): - """Just do nothing.""" - - factory.__name__ = name or factory.__name__ - return factory - - -@pytest.fixture() -def build_handler_for_collector(): - return build_botx_handler - - -@pytest.fixture() -def collector_with_handlers(build_handler_for_collector): - collector = Collector() - collector.handler(build_handler_for_collector("regular_handler")) - collector.handler( - build_handler_for_collector("regular_handler_with_command"), - command="/handler-command", - ) - collector.handler( - build_handler_for_collector("regular_handler_with_command_aliases"), - commands=["/handler-command1", "/handler-command2"], - ) - collector.handler( - build_handler_for_collector("regular_handler_with_command_and_command_aliases"), - command="/handler-command3", - commands=["/handler-command4", "/handler-command5"], - ) - collector.handler( - build_handler_for_collector("regular_handler_with_custom_name"), - name="regular-handler-with-name", - ) - collector.handler( - build_handler_for_collector("regular_handler_with_background_dependencies"), - dependencies=[Depends(build_handler_for_collector("background_dependency"))], - ) - collector.handler( - build_handler_for_collector( - "regular_handler_that_excluded_from_status_and_auto_body", - ), - include_in_status=False, - ) - collector.handler( - build_handler_for_collector( - "regular_handler_that_excluded_from_status_and_passed_body", - ), - command="regular-handler-with-excluding-from-status", - include_in_status=False, - ) - collector.handler( - build_handler_for_collector( - "regular_handler_that_included_in_status_by_callable_function", - ), - include_in_status=lambda *_: True, - ) - collector.handler( - build_handler_for_collector( - "regular_handler_that_excluded_from_status_by_callable_function", - ), - include_in_status=lambda *_: False, - ) - collector.default(build_handler_for_collector("default_handler")) - collector.hidden(build_handler_for_collector("hidden_handler")) - collector.chat_created(build_handler_for_collector("chat_created_handler")) - collector.file_transfer(build_handler_for_collector("file_transfer_handler")) - return collector diff --git a/tests/fixtures/credentials.py b/tests/fixtures/credentials.py deleted file mode 100644 index 69e5bf08..00000000 --- a/tests/fixtures/credentials.py +++ /dev/null @@ -1,23 +0,0 @@ -import uuid - -import pytest - - -@pytest.fixture() -def host(): - return "cts.example.com" - - -@pytest.fixture() -def secret_key(): - return "secret-key-for-token" - - -@pytest.fixture() -def token(): - return "generated-token-for-bot" - - -@pytest.fixture() -def bot_id(): - return uuid.uuid4() diff --git a/tests/fixtures/errors.py b/tests/fixtures/errors.py deleted file mode 100644 index 840f6067..00000000 --- a/tests/fixtures/errors.py +++ /dev/null @@ -1,18 +0,0 @@ -import threading - -import pytest - -from botx import Message - - -@pytest.fixture() -def build_exception_catcher(storage): - def factory(event: threading.Event): - def decorator(exc: Exception, msg: Message): - event.set() - storage.exception = exc - storage.message = msg - - return decorator - - return factory diff --git a/tests/fixtures/handlers.py b/tests/fixtures/handlers.py deleted file mode 100644 index ec916dbc..00000000 --- a/tests/fixtures/handlers.py +++ /dev/null @@ -1,28 +0,0 @@ -import threading -from typing import Callable - -import pytest - - -def build_botx_handler(event: threading.Event) -> Callable[[], None]: - def factory(): - event.set() - - return factory - - -@pytest.fixture() -def build_handler(): - return build_botx_handler - - -@pytest.fixture() -def build_failed_handler(): - def factory(exception: Exception, event: threading.Event): - def decorator(): - event.set() - raise exception - - return decorator - - return factory diff --git a/tests/fixtures/logging.py b/tests/fixtures/logging.py deleted file mode 100644 index 03769385..00000000 --- a/tests/fixtures/logging.py +++ /dev/null @@ -1,20 +0,0 @@ -import logging - -import pytest -from loguru import logger - - -@pytest.fixture() -def _enable_logger(): - logger.enable("botx") - - -@pytest.fixture() -def loguru_caplog(caplog, _enable_logger): - class PropogateHandler(logging.Handler): - def emit(self, record: logging.LogRecord) -> None: - logging.getLogger(record.name).handle(record) - - handler_id = logger.add(PropogateHandler(), format="{message}") - yield caplog - logger.remove(handler_id) diff --git a/tests/fixtures/messages.py b/tests/fixtures/messages.py deleted file mode 100644 index 21152cfd..00000000 --- a/tests/fixtures/messages.py +++ /dev/null @@ -1,72 +0,0 @@ -import pytest - -from botx import ( - ChatCreatedEvent, - InternalBotNotificationEvent, - InternalBotNotificationPayload, - Message, - MessageBuilder, - UserKinds, -) -from botx.models.events import UserInChatCreated - - -@pytest.fixture() -def incoming_message(host, bot_id): - builder = MessageBuilder() - builder.bot_id = bot_id - builder.user.host = host - return builder.message - - -@pytest.fixture() -def message(incoming_message, bot): - return Message.from_dict(incoming_message.dict(), bot) - - -@pytest.fixture() -def chat_created_message(host, bot_id): - builder = MessageBuilder() - builder.bot_id = bot_id - builder.command_data = ChatCreatedEvent( - group_chat_id=builder.user.group_chat_id, - chat_type=builder.user.chat_type, - name="chat", - creator=builder.user.user_huid, - members=[ - UserInChatCreated( - huid=builder.user.user_huid, - user_kind=UserKinds.user, - name=builder.user.username, - admin=True, - ), - UserInChatCreated( - huid=builder.bot_id, - user_kind=UserKinds.bot, - name="bot", - admin=False, - ), - ], - ) - builder.user.user_huid = None - builder.user.ad_login = None - builder.user.ad_domain = None - builder.user.username = None - - builder.body = "system:chat_created" - builder.system_command = True - - return builder.message - - -@pytest.fixture() -def internal_bot_notification_message(host, bot_id, bot): - builder = MessageBuilder() - builder.bot_id = bot_id - builder.command_data = InternalBotNotificationEvent( - data=InternalBotNotificationPayload(message="ping"), # noqa: WPS110 - opts={}, - ) - builder.body = "system:internal_bot_notification" - builder.system_command = True - return Message.from_dict(builder.message.dict(), bot) diff --git a/tests/fixtures/smartapps.py b/tests/fixtures/smartapps.py deleted file mode 100644 index 00763dae..00000000 --- a/tests/fixtures/smartapps.py +++ /dev/null @@ -1,34 +0,0 @@ -from typing import Dict -from uuid import UUID, uuid4 - -import pytest - - -@pytest.fixture() -def smartapp_api_version() -> int: - return 1 - - -@pytest.fixture() -def smartapp_counter() -> int: - return 42 - - -@pytest.fixture() -def smartapp_id() -> UUID: - return uuid4() - - -@pytest.fixture() -def group_chat_id() -> UUID: - return uuid4() - - -@pytest.fixture() -def ref() -> UUID: - return uuid4() - - -@pytest.fixture() -def smartapp_data() -> Dict[str, str]: - return {"key": "value"} diff --git a/tests/fixtures/storage.py b/tests/fixtures/storage.py deleted file mode 100644 index bfb2ef67..00000000 --- a/tests/fixtures/storage.py +++ /dev/null @@ -1,8 +0,0 @@ -import pytest - -from botx.models.datastructures import State - - -@pytest.fixture() -def storage(): - return State() diff --git a/tests/test_clients/test_methods/test_base/__init__.py b/tests/models/__init__.py similarity index 100% rename from tests/test_clients/test_methods/test_base/__init__.py rename to tests/models/__init__.py diff --git a/tests/models/test_incoming_message.py b/tests/models/test_incoming_message.py new file mode 100644 index 00000000..081848aa --- /dev/null +++ b/tests/models/test_incoming_message.py @@ -0,0 +1,101 @@ +from typing import Callable, Tuple + +import pytest + +from botx import IncomingMessage + + +def test__upn_property__not_filled( + incoming_message_factory: Callable[..., IncomingMessage], +) -> None: + # - Arrange - + message = incoming_message_factory() + + # - Assert - + assert message.sender.upn is None + + +def test__upn_property__filled( + incoming_message_factory: Callable[..., IncomingMessage], +) -> None: + # - Arrange - + message = incoming_message_factory(ad_login="login", ad_domain="domain") + + # - Assert - + assert message.sender.upn == "login@domain" + + +@pytest.mark.parametrize( + "body,argument_answer", + [ + ("", ""), + ("/command", ""), + ], +) +def test__argument__not_filled( + incoming_message_factory: Callable[..., IncomingMessage], + body: str, + argument_answer: str, +) -> None: + # - Arrange - + message = incoming_message_factory(body=body) + + # - Assert - + assert message.argument == argument_answer + + +@pytest.mark.parametrize( + "body,argument_answer", + [ + ("/command arg1 ", "arg1"), + ("/command arg1 arg2", "arg1 arg2"), + ], +) +def test__argument__filled( + incoming_message_factory: Callable[..., IncomingMessage], + body: str, + argument_answer: str, +) -> None: + # - Arrange - + message = incoming_message_factory(body=body) + + # - Assert - + assert message.argument == argument_answer + + +@pytest.mark.parametrize( + "body,argument_answer", + [ + ("", ()), + ("/command", ()), + ], +) +def test__arguments__not_filled( + incoming_message_factory: Callable[..., IncomingMessage], + body: str, + argument_answer: Tuple[str, ...], +) -> None: + # - Arrange - + message = incoming_message_factory(body=body) + + # - Assert - + assert message.arguments == argument_answer + + +@pytest.mark.parametrize( + "body,argument_answer", + [ + ("/command arg1 ", ("arg1",)), + ("/command arg1 arg2", ("arg1", "arg2")), + ], +) +def test__arguments__filled( + incoming_message_factory: Callable[..., IncomingMessage], + body: str, + argument_answer: Tuple[str, ...], +) -> None: + # - Arrange - + message = incoming_message_factory(body=body) + + # - Assert - + assert message.arguments == argument_answer diff --git a/tests/models/test_mentions_list.py b/tests/models/test_mentions_list.py new file mode 100644 index 00000000..17cf5707 --- /dev/null +++ b/tests/models/test_mentions_list.py @@ -0,0 +1,88 @@ +from typing import Callable, Optional +from uuid import UUID, uuid4 + +import pytest + +from botx import Mention, MentionList, MentionTypes + + +@pytest.fixture +def mention_factory() -> Callable[..., Mention]: + def factory( + mention_type: MentionTypes, + huid: UUID, + name: Optional[str] = None, + ) -> Mention: + return Mention( + type=mention_type, + entity_id=huid, + name=name, + ) + + return factory + + +def test__mentions_list_properties__filled( + mention_factory: Callable[..., Mention], +) -> None: + # - Arrange - + contacts = [ + mention_factory( + mention_type=MentionTypes.CONTACT, + huid=uuid4(), + name=str(name), + ) + for name in range(2) + ] + chats = [ + mention_factory( + mention_type=MentionTypes.CHAT, + huid=uuid4(), + name=str(name), + ) + for name in range(2) + ] + channels = [ + mention_factory( + mention_type=MentionTypes.CHANNEL, + huid=uuid4(), + name=str(name), + ) + for name in range(2) + ] + users = [ + mention_factory( + mention_type=MentionTypes.USER, + huid=uuid4(), + name=str(name), + ) + for name in range(2) + ] + + mentions = MentionList([*contacts, *chats, *channels, *users]) + + # - Assert - + assert mentions.contacts == contacts + assert mentions.chats == chats + assert mentions.channels == channels + assert mentions.users == users + + +def test__mentions_list_all_users_mentioned__filled( + mention_factory: Callable[..., Mention], +) -> None: + # - Arrange - + user_mention = mention_factory( + mention_type=MentionTypes.CONTACT, + huid=uuid4(), + ) + all_mention = Mention(type=MentionTypes.ALL) + + one_all_mention = MentionList([user_mention, all_mention]) + two_all_mentions = MentionList([all_mention, all_mention]) + + # - Assert - + assert one_all_mention.all_users_mentioned + assert two_all_mentions.all_users_mentioned + + assert not MentionList([]).all_users_mentioned diff --git a/tests/models/test_status_recipient.py b/tests/models/test_status_recipient.py new file mode 100644 index 00000000..5d3f8c2a --- /dev/null +++ b/tests/models/test_status_recipient.py @@ -0,0 +1,24 @@ +from typing import Callable + +from botx import IncomingMessage, StatusRecipient + + +def test__status_recipient__from_message( + incoming_message_factory: Callable[..., IncomingMessage], +) -> None: + # - Arrange - + incoming_message = incoming_message_factory( + ad_login="test_login", + ad_domain="test_domain", + ) + status_recipient = StatusRecipient.from_incoming_message(incoming_message) + + # - Assert - + assert status_recipient == StatusRecipient( + bot_id=incoming_message.bot.id, + huid=incoming_message.sender.huid, + ad_login=incoming_message.sender.ad_login, + ad_domain=incoming_message.sender.ad_domain, + is_admin=incoming_message.sender.is_chat_admin, + chat_type=incoming_message.chat.type, + ) diff --git a/tests/system_events/test_added_to_chat.py b/tests/system_events/test_added_to_chat.py new file mode 100644 index 00000000..6788e60e --- /dev/null +++ b/tests/system_events/test_added_to_chat.py @@ -0,0 +1,100 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + AddedToChatEvent, + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatTypes, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__added_to_chat__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:added_to_chat", + "command_type": "system", + "data": { + "added_members": [ + "ab103983-6001-44e9-889e-d55feb295494", + "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + ], + }, + "metadata": {}, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "group_chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + added_to_chat: Optional[AddedToChatEvent] = None + + @collector.added_to_chat + async def added_to_chat_handler(event: AddedToChatEvent, bot: Bot) -> None: + nonlocal added_to_chat + added_to_chat = event + # Drop `raw_command` from asserting + added_to_chat.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert added_to_chat == AddedToChatEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + huids=[ + UUID("ab103983-6001-44e9-889e-d55feb295494"), + UUID("dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4"), + ], + chat=Chat( + id=UUID("dea55ee4-7a9f-5da0-8c73-079f400ee517"), + type=ChatTypes.GROUP_CHAT, + ), + ) diff --git a/tests/system_events/test_chat_created.py b/tests/system_events/test_chat_created.py new file mode 100644 index 00000000..806d3338 --- /dev/null +++ b/tests/system_events/test_chat_created.py @@ -0,0 +1,129 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatCreatedEvent, + ChatCreatedMember, + ChatTypes, + HandlerCollector, + UserKinds, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__chat_created__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:chat_created", + "command_type": "system", + "data": { + "chat_type": "group_chat", + "creator": "83fbf1c7-f14b-5176-bd32-ca15cf00d4b7", + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "members": [ + { + "admin": True, + "huid": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "name": "Feature bot", + "user_kind": "botx", + }, + { + "admin": False, + "huid": "83fbf1c7-f14b-5176-bd32-ca15cf00d4b7", + "name": "Ivanov Ivan Ivanovich", + "user_kind": "cts_user", + }, + ], + "name": "Feature-party", + }, + "metadata": {}, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "group_chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + chat_created: Optional[ChatCreatedEvent] = None + + @collector.chat_created + async def chat_created_handler(event: ChatCreatedEvent, bot: Bot) -> None: + nonlocal chat_created + chat_created = event + # Drop `raw_command` from asserting + chat_created.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert chat_created == ChatCreatedEvent( + sync_id=UUID("2c1a31d6-f47f-5f54-aee2-d0c526bb1d54"), + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + chat_name="Feature-party", + chat=Chat( + id=UUID("dea55ee4-7a9f-5da0-8c73-079f400ee517"), + type=ChatTypes.GROUP_CHAT, + ), + creator_id=UUID("83fbf1c7-f14b-5176-bd32-ca15cf00d4b7"), + members=[ + ChatCreatedMember( + is_admin=True, + huid=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + username="Feature bot", + kind=UserKinds.BOT, + ), + ChatCreatedMember( + is_admin=False, + huid=UUID("83fbf1c7-f14b-5176-bd32-ca15cf00d4b7"), + username="Ivanov Ivan Ivanovich", + kind=UserKinds.CTS_USER, + ), + ], + raw_command=None, + ) diff --git a/tests/system_events/test_cts_login.py b/tests/system_events/test_cts_login.py new file mode 100644 index 00000000..5c3effef --- /dev/null +++ b/tests/system_events/test_cts_login.py @@ -0,0 +1,89 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + CTSLoginEvent, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__cts_login__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:cts_login", + "data": { + "user_huid": "b9197d3a-d855-5d34-ba8a-eff3a975ab20", + "cts_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + "command_type": "system", + "metadata": {}, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": None, + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": None, + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + cts_login: Optional[CTSLoginEvent] = None + + @collector.cts_login + async def cts_login_handler(event: CTSLoginEvent, bot: Bot) -> None: + nonlocal cts_login + cts_login = event + # Drop `raw_command` from asserting + cts_login.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert cts_login == CTSLoginEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + huid=UUID("b9197d3a-d855-5d34-ba8a-eff3a975ab20"), + ) diff --git a/tests/system_events/test_cts_logout.py b/tests/system_events/test_cts_logout.py new file mode 100644 index 00000000..13363bdc --- /dev/null +++ b/tests/system_events/test_cts_logout.py @@ -0,0 +1,89 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + CTSLogoutEvent, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__cts_logout__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:cts_logout", + "data": { + "user_huid": "b9197d3a-d855-5d34-ba8a-eff3a975ab20", + "cts_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + "command_type": "system", + "metadata": {}, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": None, + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": None, + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + cts_logout: Optional[CTSLogoutEvent] = None + + @collector.cts_logout + async def cts_logout_handler(event: CTSLogoutEvent, bot: Bot) -> None: + nonlocal cts_logout + cts_logout = event + # Drop `raw_command` from asserting + cts_logout.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert cts_logout == CTSLogoutEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + huid=UUID("b9197d3a-d855-5d34-ba8a-eff3a975ab20"), + ) diff --git a/tests/system_events/test_deleted_from_chat.py b/tests/system_events/test_deleted_from_chat.py new file mode 100644 index 00000000..4f93fa30 --- /dev/null +++ b/tests/system_events/test_deleted_from_chat.py @@ -0,0 +1,99 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatTypes, + DeletedFromChatEvent, + HandlerCollector, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__deleted_from_chat__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:deleted_from_chat", + "command_type": "system", + "data": { + "deleted_members": [ + "ab103983-6001-44e9-889e-d55feb295494", + "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + ], + }, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "group_chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + deleted_from_chat: Optional[DeletedFromChatEvent] = None + + @collector.deleted_from_chat + async def deleted_from_chat_handler(event: DeletedFromChatEvent, bot: Bot) -> None: + nonlocal deleted_from_chat + deleted_from_chat = event + # Drop `raw_command` from asserting + deleted_from_chat.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert deleted_from_chat == DeletedFromChatEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + huids=[ + UUID("ab103983-6001-44e9-889e-d55feb295494"), + UUID("dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4"), + ], + chat=Chat( + id=UUID("dea55ee4-7a9f-5da0-8c73-079f400ee517"), + type=ChatTypes.GROUP_CHAT, + ), + ) diff --git a/tests/system_events/test_internal_bot_notification.py b/tests/system_events/test_internal_bot_notification.py new file mode 100644 index 00000000..f39b636f --- /dev/null +++ b/tests/system_events/test_internal_bot_notification.py @@ -0,0 +1,108 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + BotSender, + Chat, + ChatTypes, + HandlerCollector, + InternalBotNotificationEvent, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__internal_bot_notification__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "sync_id": "a465f0f3-1354-491c-8f11-f400164295cb", + "command": { + "body": "system:internal_bot_notification", + "data": { + "data": { + "message": "ping", + }, + "opts": { + "internal_token": "KyKfLJD1zMjNSJ1cQ4+8Lz", + }, + }, + "command_type": "system", + "metadata": {}, + }, + "async_files": [], + "attachments": [], + "entities": [], + "from": { + "user_huid": "b9197d3a-d855-5d34-ba8a-eff3a975ab20", + "group_chat_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "ad_login": None, + "ad_domain": None, + "username": None, + "chat_type": "group_chat", + "manufacturer": None, + "device": None, + "device_software": None, + "device_meta": {}, + "platform": None, + "platform_package_id": None, + "is_admin": False, + "is_creator": False, + "app_version": None, + "locale": "en", + "host": "cts.example.com", + }, + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "proto_version": 4, + "source_sync_id": None, + } + + collector = HandlerCollector() + internal_bot_notification: Optional[InternalBotNotificationEvent] = None + + @collector.internal_bot_notification + async def internal_bot_notification_handler( + event: InternalBotNotificationEvent, + bot: Bot, + ) -> None: + nonlocal internal_bot_notification + internal_bot_notification = event + # Drop `raw_command` from asserting + internal_bot_notification.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert internal_bot_notification == InternalBotNotificationEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + data={"message": "ping"}, + opts={"internal_token": "KyKfLJD1zMjNSJ1cQ4+8Lz"}, + chat=Chat( + id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + type=ChatTypes.GROUP_CHAT, + ), + sender=BotSender( + huid=UUID("b9197d3a-d855-5d34-ba8a-eff3a975ab20"), + is_chat_admin=False, + is_chat_creator=False, + ), + ) diff --git a/tests/system_events/test_left_from_chat.py b/tests/system_events/test_left_from_chat.py new file mode 100644 index 00000000..918ce34b --- /dev/null +++ b/tests/system_events/test_left_from_chat.py @@ -0,0 +1,99 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatTypes, + HandlerCollector, + LeftFromChatEvent, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__left_from_chat__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "system:left_from_chat", + "command_type": "system", + "data": { + "left_members": [ + "ab103983-6001-44e9-889e-d55feb295494", + "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", + ], + }, + }, + "source_sync_id": None, + "sync_id": "2c1a31d6-f47f-5f54-aee2-d0c526bb1d54", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "group_chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": None, + "timezone": None, + }, + "device_software": None, + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "host": "cts.example.com", + "is_admin": None, + "is_creator": None, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": None, + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + left_from_chat: Optional[LeftFromChatEvent] = None + + @collector.left_from_chat + async def left_from_chat_handler(event: LeftFromChatEvent, bot: Bot) -> None: + nonlocal left_from_chat + left_from_chat = event + # Drop `raw_command` from asserting + left_from_chat.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert left_from_chat == LeftFromChatEvent( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + raw_command=None, + huids=[ + UUID("ab103983-6001-44e9-889e-d55feb295494"), + UUID("dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4"), + ], + chat=Chat( + id=UUID("dea55ee4-7a9f-5da0-8c73-079f400ee517"), + type=ChatTypes.GROUP_CHAT, + ), + ) diff --git a/tests/system_events/test_smartapp_event.py b/tests/system_events/test_smartapp_event.py new file mode 100644 index 00000000..72dcabb4 --- /dev/null +++ b/tests/system_events/test_smartapp_event.py @@ -0,0 +1,163 @@ +from typing import Optional +from uuid import UUID + +import pytest + +from botx import ( + AttachmentTypes, + Bot, + BotAccount, + BotAccountWithSecret, + HandlerCollector, + Image, + SmartAppEvent, + lifespan_wrapper, +) +from botx.models.chats import Chat +from botx.models.enums import ChatTypes +from botx.models.message.incoming_message import UserDevice, UserSender + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__smartapp__succeed( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "sync_id": "a465f0f3-1354-491c-8f11-f400164295cb", + "command": { + "body": "system:smartapp_event", + "data": { + "ref": "6fafda2c-6505-57a5-a088-25ea5d1d0364", + "smartapp_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "data": { + "type": "smartapp_rpc", + "method": "folders.get", + "params": { + "q": 1, + }, + }, + "opts": {"option": "test_option"}, + "smartapp_api_version": 1, + }, + "command_type": "system", + "metadata": {}, + }, + "async_files": [ + { + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + ], + "attachments": [], + "entities": [], + "from": { + "user_huid": "b9197d3a-d855-5d34-ba8a-eff3a975ab20", + "group_chat_id": "dea55ee4-7a9f-5da0-8c73-079f400ee517", + "host": "cts.example.com", + "ad_login": None, + "ad_domain": None, + "username": None, + "chat_type": "group_chat", + "manufacturer": None, + "device": None, + "device_software": None, + "device_meta": {}, + "platform": None, + "platform_package_id": None, + "is_admin": False, + "is_creator": False, + "app_version": None, + "locale": "en", + }, + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "proto_version": 4, + "source_sync_id": None, + } + + collector = HandlerCollector() + smartapp: Optional[SmartAppEvent] = None + + @collector.smartapp_event + async def smartapp_handler(event: SmartAppEvent, bot: Bot) -> None: + nonlocal smartapp + smartapp = event + # Drop `raw_command` from asserting + smartapp.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert smartapp == SmartAppEvent( + ref=UUID("6fafda2c-6505-57a5-a088-25ea5d1d0364"), + smartapp_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + data={ + "type": "smartapp_rpc", + "method": "folders.get", + "params": { + "q": 1, + }, + }, + opts={"option": "test_option"}, + smartapp_api_version=1, + files=[ + Image( + type=AttachmentTypes.IMAGE, + filename="pass.png", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="image/png", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ], + chat=Chat( + id=UUID("dea55ee4-7a9f-5da0-8c73-079f400ee517"), + type=ChatTypes.GROUP_CHAT, + ), + sender=UserSender( + huid=UUID("b9197d3a-d855-5d34-ba8a-eff3a975ab20"), + ad_login=None, + ad_domain=None, + username=None, + is_chat_admin=False, + is_chat_creator=False, + device=UserDevice( + manufacturer=None, + device_name=None, + os=None, + pushes=None, + timezone=None, + permissions=None, + platform=None, + platform_package_id=None, + app_version=None, + locale="en", + ), + ), + raw_command=None, + ) diff --git a/tests/test_attachments.py b/tests/test_attachments.py new file mode 100644 index 00000000..add65590 --- /dev/null +++ b/tests/test_attachments.py @@ -0,0 +1,272 @@ +import asyncio +from typing import Any, Callable, Dict, Optional +from uuid import UUID + +import pytest + +from botx import ( + AttachmentTypes, + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + lifespan_wrapper, +) +from botx.models.attachments import ( + AttachmentContact, + AttachmentDocument, + AttachmentImage, + AttachmentLink, + AttachmentLocation, + AttachmentVideo, + AttachmentVoice, + IncomingAttachment, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__attachment__open( + host: str, + bot_account: BotAccountWithSecret, + bot_id: UUID, + api_incoming_message_factory: Callable[..., Dict[str, Any]], +) -> None: + # - Arrange - + payload = api_incoming_message_factory( + bot_id=bot_id, + attachment={ + "data": { + "content": "", + "file_name": "test_file.jpg", + }, + "type": "image", + }, + group_chat_id="054af49e-5e18-4dca-ad73-4f96b6de63fa", + host=host, + ) + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + await asyncio.sleep(0) # Return control to event loop + + assert incoming_message and incoming_message.file + async with incoming_message.file.open() as fo: + read_content = await fo.read() + + # - Assert - + assert read_content == b"Hello, world!\n" + + +API_AND_DOMAIN_NON_FILE_ATTACHMENTS = ( + ( + { + "type": "location", + "data": { + "location_name": "Центр вселенной", + "location_address": "Россия, Тверская область", + "location_lat": 58.04861, + "location_lng": 34.28833, + }, + }, + AttachmentLocation( + type=AttachmentTypes.LOCATION, + name="Центр вселенной", + address="Россия, Тверская область", + latitude="58.04861", + longitude="34.28833", + ), + "location", + ), + ( + { + "type": "contact", + "data": { + "file_name": "Контакт", + "contact_name": "Иванов Иван", + "content": "data:text/vcard;base64,eDnXAc1FEUB0VFEFctII3lRlRBcetROeFfduPmXxE/8=", + }, + }, + AttachmentContact( + type=AttachmentTypes.CONTACT, + name="Иванов Иван", + ), + "contact", + ), + ( + { + "type": "link", + "data": { + "url": "http://ya.ru/xxx", + "url_title": "Header in link", + "url_preview": "http://ya.ru/xxx.jpg", + "url_text": "Some text in link", + }, + }, + AttachmentLink( + type=AttachmentTypes.LINK, + url="http://ya.ru/xxx", + title="Header in link", + preview="http://ya.ru/xxx.jpg", + text="Some text in link", + ), + "link", + ), +) + + +@pytest.mark.parametrize( + "api_attachment,domain_attachment,attr_name", + API_AND_DOMAIN_NON_FILE_ATTACHMENTS, +) +async def test__async_execute_raw_bot_command__non_file_attachments_types( + api_attachment: Dict[str, Any], + domain_attachment: IncomingAttachment, + attr_name: str, + api_incoming_message_factory: Callable[..., Dict[str, Any]], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = api_incoming_message_factory(attachment=api_attachment) + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert getattr(incoming_message, attr_name) == domain_attachment + + +API_AND_DOMAIN_FILE_ATTACHMENTS = ( + ( + { + "data": { + "content": "", + "file_name": "test_file.jpg", + }, + "type": "image", + }, + AttachmentImage( + type=AttachmentTypes.IMAGE, + filename="test_file.jpg", + size=len(b"Hello, world!\n"), + is_async_file=False, + content=b"Hello, world!\n", + ), + ), + ( + { + "data": { + "content": "data:video/mp4;base64,SGVsbG8sIHdvcmxkIQo=", + "file_name": "test_file.mp4", + "duration": 10, + }, + "type": "video", + }, + AttachmentVideo( + type=AttachmentTypes.VIDEO, + filename="test_file.mp4", + size=len(b"Hello, world!\n"), + is_async_file=False, + content=b"Hello, world!\n", + duration=10, + ), + ), + ( + { + "data": { + "content": "data:text/plain;base64,SGVsbG8sIHdvcmxkIQo=", + "file_name": "test_file.txt", + }, + "type": "document", + }, + AttachmentDocument( + type=AttachmentTypes.DOCUMENT, + filename="test_file.txt", + size=len(b"Hello, world!\n"), + is_async_file=False, + content=b"Hello, world!\n", + ), + ), + ( + { + "data": { + "content": "data:audio/mpeg3;base64,SGVsbG8sIHdvcmxkIQo=", + "duration": 10, + }, + "type": "voice", + }, + AttachmentVoice( + type=AttachmentTypes.VOICE, + filename="record.mp3", + size=len(b"Hello, world!\n"), + is_async_file=False, + content=b"Hello, world!\n", + duration=10, + ), + ), +) + + +@pytest.mark.parametrize( + "api_attachment,domain_attachment", + API_AND_DOMAIN_FILE_ATTACHMENTS, +) +async def test__async_execute_raw_bot_command__file_attachments_types( + api_attachment: Dict[str, Any], + domain_attachment: IncomingAttachment, + api_incoming_message_factory: Callable[..., Dict[str, Any]], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = api_incoming_message_factory(attachment=api_attachment) + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert incoming_message + assert incoming_message.file == domain_attachment diff --git a/tests/test_base_command.py b/tests/test_base_command.py new file mode 100644 index 00000000..2e7086f6 --- /dev/null +++ b/tests/test_base_command.py @@ -0,0 +1,38 @@ +import pytest + +from botx import Bot, HandlerCollector, UnsupportedBotAPIVersionError, lifespan_wrapper + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__async_execute_raw_bot_command__invalid_payload_value_error_raised() -> None: + # - Arrange - + payload = {"invalid": "command"} + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(ValueError) as exc: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert "validation" in str(exc.value) + + +async def test__async_execute_raw_bot_command__unsupported_bot_api_version_error_raised() -> None: + # - Arrange - + payload = {"proto_version": "3"} + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UnsupportedBotAPIVersionError) as exc: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert "Unsupported" in str(exc.value) + assert "expected `4`" in str(exc.value) diff --git a/tests/test_bot_constructing.py b/tests/test_bot_constructing.py new file mode 100644 index 00000000..41ebedd0 --- /dev/null +++ b/tests/test_bot_constructing.py @@ -0,0 +1,24 @@ +import pytest + +from botx import Bot, BotAccountWithSecret, HandlerCollector + + +def test__bot__empty_collectors_warning( + loguru_caplog: pytest.LogCaptureFixture, + bot_account: BotAccountWithSecret, +) -> None: + # - Act - + Bot(collectors=[], bot_accounts=[bot_account]) + + # - Assert - + assert "Bot has no connected collectors" in loguru_caplog.text + + +def test__bot__empty_bot_accounts_warning( + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Act - + Bot(collectors=[HandlerCollector()], bot_accounts=[]) + + # - Assert - + assert "Bot has no bot accounts" in loguru_caplog.text diff --git a/tests/test_bots/test_bots/test_decorators/test_exception_handler.py b/tests/test_bots/test_bots/test_decorators/test_exception_handler.py deleted file mode 100644 index ff8ecdad..00000000 --- a/tests/test_bots/test_bots/test_decorators/test_exception_handler.py +++ /dev/null @@ -1,21 +0,0 @@ -import threading - -import pytest - -pytestmark = pytest.mark.asyncio - - -async def test_register_middleware_through_decorator( - bot, - client, - incoming_message, - build_failed_handler, - build_exception_catcher, -): - exception_event = threading.Event() - bot.exception_handler(Exception)(build_exception_catcher(exception_event)) - bot.default(build_failed_handler(Exception(), threading.Event())) - - await client.send_command(incoming_message) - - assert exception_event.is_set() diff --git a/tests/test_bots/test_bots/test_decorators/test_middleware.py b/tests/test_bots/test_bots/test_decorators/test_middleware.py deleted file mode 100644 index 390fde05..00000000 --- a/tests/test_bots/test_bots/test_decorators/test_middleware.py +++ /dev/null @@ -1,17 +0,0 @@ -import pytest - -from botx import Message -from botx.typing import SyncExecutor - - -@pytest.fixture() -def middleware_function(): - def factory(_message: Message, _call_next: SyncExecutor): - """Do nothing.""" - - return factory - - -def test_register_middleware_through_decorator(bot, middleware_function): - bot.middleware(middleware_function) - assert bot.exception_middleware.executor.dispatch_func == middleware_function diff --git a/tests/test_bots/test_bots/test_execution.py b/tests/test_bots/test_bots/test_execution.py deleted file mode 100644 index db0f9f38..00000000 --- a/tests/test_bots/test_bots/test_execution.py +++ /dev/null @@ -1,24 +0,0 @@ -import threading - -import pytest - -pytestmark = pytest.mark.asyncio - - -async def test_bot_process_message_by_sorted_handlers( - bot, - client, - incoming_message, - build_handler, -): - event1 = threading.Event() - event2 = threading.Event() - - bot.handler(build_handler(event1), command="/body") - bot.handler(build_handler(event2), command="/body-v2") - - incoming_message.command.body = "/body-v2 args" - await client.send_command(incoming_message) - - assert not event1.is_set() - assert event2.is_set() diff --git a/tests/test_bots/test_bots/test_execution_errors.py b/tests/test_bots/test_bots/test_execution_errors.py deleted file mode 100644 index d93c6d35..00000000 --- a/tests/test_bots/test_bots/test_execution_errors.py +++ /dev/null @@ -1,29 +0,0 @@ -import uuid - -import pytest - -from botx import BotXCredentials, UnknownBotError - -pytestmark = pytest.mark.asyncio - - -async def test_error_when_execution_for_unknown_host( - bot, - client, - incoming_message, - bot_id, -): - bot.bot_accounts = [ - BotXCredentials( - host="cts.unknown1.com", - secret_key="secret", - bot_id=uuid.uuid4(), - ), - BotXCredentials( - host="cts.unknown2.com", - secret_key="secret", - bot_id=uuid.uuid4(), - ), - ] - with pytest.raises(UnknownBotError): - await client.send_command(incoming_message) diff --git a/tests/test_bots/test_bots/test_lifespan.py b/tests/test_bots/test_bots/test_lifespan.py deleted file mode 100644 index ee04fd83..00000000 --- a/tests/test_bots/test_bots/test_lifespan.py +++ /dev/null @@ -1,38 +0,0 @@ -import threading -from typing import Callable - -import pytest - -from botx import Bot - -pytestmark = pytest.mark.asyncio - - -def build_lifespan_event(event: threading.Event) -> Callable[[Bot], None]: - def factory(_bot): - event.set() - - return factory - - -@pytest.fixture() -def build_lifespan(): - return build_lifespan_event - - -async def test_lifespan_events(bot, build_lifespan): - startup_event = threading.Event() - shutdown_event = threading.Event() - - bot.startup_events = [build_lifespan(startup_event)] - bot.shutdown_events = [build_lifespan(shutdown_event)] - - await bot.start() - assert startup_event.is_set() - - await bot.shutdown() - assert shutdown_event.is_set() - - -async def test_no_error_when_stopping_bot_with_no_tasks(bot): - await bot.shutdown() diff --git a/tests/test_bots/test_bots/test_status.py b/tests/test_bots/test_bots/test_status.py deleted file mode 100644 index 2bbef91d..00000000 --- a/tests/test_bots/test_bots/test_status.py +++ /dev/null @@ -1,21 +0,0 @@ -import pytest - -pytestmark = pytest.mark.asyncio - - -async def test_returning_bot_commands_status(bot, collector_with_handlers): - bot.include_collector(collector_with_handlers) - status = await bot.status() - commands = [command.body for command in status.result.commands] - assert commands == [ - "/regular-handler", - "/handler-command", - "/handler-command1", - "/handler-command2", - "/handler-command3", - "/handler-command4", - "/handler-command5", - "/regular-handler-with-name", - "/regular-handler-with-background-dependencies", - "/regular-handler-that-included-in-status-by-callable-function", - ] diff --git a/tests/test_bots/test_mixins/test_clients.py b/tests/test_bots/test_mixins/test_clients.py deleted file mode 100644 index d918ad0f..00000000 --- a/tests/test_bots/test_mixins/test_clients.py +++ /dev/null @@ -1,38 +0,0 @@ -import uuid - -import pytest - -from botx import BotXCredentials -from botx.exceptions import TokenError, UnknownBotError - -pytestmark = pytest.mark.asyncio - - -def test_raising_error_if_token_was_not_found(client, incoming_message): - account = client.bot.get_account_by_bot_id(incoming_message.bot_id) - account.token = None - with pytest.raises(TokenError): - client.bot.get_token_for_bot(incoming_message.bot_id) - - -def test_get_token_to_bot(client, incoming_message): - account = client.bot.get_account_by_bot_id(incoming_message.bot_id) - account.token = "token" - assert client.bot.get_token_for_bot(incoming_message.bot_id) is not None - - -def test_raising_error_if_cts_not_found(bot, incoming_message): - bot.bot_accounts = [ - BotXCredentials( - host="cts.unknown1.com", - secret_key="secret", - bot_id=uuid.uuid4(), - ), - BotXCredentials( - host="cts.unknown2.com", - secret_key="secret", - bot_id=uuid.uuid4(), - ), - ] - with pytest.raises(UnknownBotError): - bot.get_account_by_bot_id(incoming_message.bot_id) diff --git a/tests/test_bots/test_mixins/test_requests/test_chats.py b/tests/test_bots/test_mixins/test_requests/test_chats.py deleted file mode 100644 index 71244fa7..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_chats.py +++ /dev/null @@ -1,121 +0,0 @@ -import uuid - -import pytest - -from botx import ChatTypes -from botx.clients.methods.v3.chats.chat_list import ChatList - -pytestmark = pytest.mark.asyncio - - -async def test_creating_chat(client, message): - await client.bot.create_chat( - message.credentials, - name="test", - members=[message.user_huid], - chat_type=ChatTypes.group_chat, - ) - - assert client.requests[0].name == "test" - - -async def test_enable_stealth_mode(bot, client, message): - await bot.enable_stealth_mode( - message.credentials, - chat_id=message.group_chat_id, - burn_in=60, - ) - - assert client.requests[0].burn_in == 60 - - -async def test_disable_stealth_mode(bot, client, message): - await bot.disable_stealth_mode( - message.credentials, - chat_id=message.group_chat_id, - ) - - assert client.requests[0].group_chat_id == message.group_chat_id - - -async def test_adding_user_to_chat(bot, client, message): - users = [uuid.uuid4()] - await bot.add_users( - message.credentials, - chat_id=message.group_chat_id, - user_huids=users, - ) - request = client.requests[0] - - assert request.group_chat_id == message.group_chat_id - - assert request.user_huids == users - - -async def test_remove_user(bot, client, message): - users = [uuid.uuid4()] - await bot.remove_users( - message.credentials, - chat_id=message.group_chat_id, - user_huids=users, - ) - request = client.requests[0] - - assert request.group_chat_id == message.group_chat_id - - assert request.user_huids == users - - -async def test_retrieving_chat_info(bot, client, message): - chat_id = uuid.uuid4() - info = await bot.get_chat_info(message.credentials, chat_id=chat_id) - - assert info.group_chat_id == chat_id - - -async def test_retrieving_bot_chats(bot, client, message): - await bot.get_bot_chats(message.credentials) - request = client.requests[0] - - assert isinstance(request, ChatList) - - -async def test_promoting_users_to_admins(bot, client, message): - users = [uuid.uuid4()] - await bot.add_admin_roles( - message.credentials, - chat_id=message.group_chat_id, - user_huids=users, - ) - request = client.requests[0] - - assert request.group_chat_id == message.group_chat_id - - assert request.user_huids == users - - -async def test_pinning_message(bot, client, message): - chat_id = uuid.uuid4() - sync_id = uuid.uuid4() - - await bot.pin_message( - message.credentials, - chat_id=chat_id, - sync_id=sync_id, - ) - request = client.requests[0] - - assert request.chat_id == chat_id - assert request.sync_id == sync_id - - -async def test_unpinning_message(bot, client, message): - chat_id = uuid.uuid4() - - await bot.unpin_message( - message.credentials, - chat_id=chat_id, - ) - request = client.requests[0] - - assert request.chat_id == chat_id diff --git a/tests/test_bots/test_mixins/test_requests/test_command.py b/tests/test_bots/test_mixins/test_requests/test_command.py deleted file mode 100644 index ce94e37c..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_command.py +++ /dev/null @@ -1,14 +0,0 @@ -import pytest - -from botx import MessagePayload - -pytestmark = pytest.mark.asyncio - - -async def test_command_result(client, message): - await client.bot.send_command_result( - credentials=message.credentials, - payload=MessagePayload(text="some text"), - ) - - assert client.command_results[0].result.body == "some text" diff --git a/tests/test_bots/test_mixins/test_requests/test_events.py b/tests/test_bots/test_mixins/test_requests/test_events.py deleted file mode 100644 index 7c70aea0..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_events.py +++ /dev/null @@ -1,69 +0,0 @@ -import pytest - -from botx import SendingMessage, UpdatePayload - -pytestmark = pytest.mark.asyncio - - -async def test_updating_message_through_bot(bot, client, message): - sync_id = await bot.answer_message("some text", message) - - await bot.update_message( - message.credentials.copy(update={"sync_id": sync_id}), - UpdatePayload(text="new text"), - ) - - update = client.message_updates[0].result - assert update.body == "new text" - - -async def test_update_metadata(bot, client, message): - msg = SendingMessage.from_message(text="some text", message=message) - msg.metadata = {"hello": "world"} - await bot.send(msg) - - upd = UpdatePayload.from_sending_payload(msg.payload) - upd.metadata = {"foo": "bar"} - - await bot.update_message(message.credentials, upd) - - update = client.message_updates[0].result - assert update.metadata == {"foo": "bar"} - - -async def test_cant_update_without_sync_id(bot, client, message): - credentials = message.credentials.copy(update={"sync_id": None}) - - with pytest.raises(ValueError) as exc: - await bot.update_message(credentials, UpdatePayload(text="new text")) - - assert "sync_id is required" in str(exc.value) - - -async def test_reply(bot, client, message): - await bot.reply( - text="foo", - source_sync_id=message.sync_id, - credentials=message.credentials, - ) - - reply = client.replies[0] - assert reply.result.body == "foo" - assert reply.source_sync_id == message.sync_id - - -async def test_reply_on_message_empty_text_error(bot, message): - with pytest.raises(ValueError): - await bot.reply( - text="", - source_sync_id=message.sync_id, - credentials=message.credentials, - ) - - -async def test_reply_arguments_error(bot, message): - with pytest.raises(ValueError): - await bot.reply( - source_sync_id=message.sync_id, - credentials=message.credentials, - ) diff --git a/tests/test_bots/test_mixins/test_requests/test_files.py b/tests/test_bots/test_mixins/test_requests/test_files.py deleted file mode 100644 index cccdda76..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_files.py +++ /dev/null @@ -1,38 +0,0 @@ -from uuid import uuid4 - -import pytest - -from botx.clients.methods.v3.files.download import DownloadFile -from botx.clients.methods.v3.files.upload import UploadFile -from botx.models.files import File -from botx.testing.content import PNG_DATA - -pytestmark = pytest.mark.asyncio - - -async def test_upload_file(client, message): - image = File(file_name="image.png", data=PNG_DATA) - await client.bot.upload_file(message.credentials, image, group_chat_id=uuid4()) - - assert isinstance(client.requests[0], UploadFile) - - -async def test_download_file(client, message): - await client.bot.download_file( - message.credentials, - file_id=uuid4(), - group_chat_id=uuid4(), - ) - - assert isinstance(client.requests[0], DownloadFile) - - -async def test_custom_filename(client, message): - file = await client.bot.download_file( - message.credentials, - file_id=uuid4(), - group_chat_id=uuid4(), - file_name="myname", - ) - - assert file.file_name == "myname.txt" diff --git a/tests/test_bots/test_mixins/test_requests/test_internal_bot_notification.py b/tests/test_bots/test_mixins/test_requests/test_internal_bot_notification.py deleted file mode 100644 index 92cab4cf..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_internal_bot_notification.py +++ /dev/null @@ -1,18 +0,0 @@ -import uuid - -import pytest - -pytestmark = pytest.mark.asyncio - - -async def test_internal_bot_notification(client, message): - await client.bot.internal_bot_notification( - credentials=message.credentials, - group_chat_id=uuid.uuid4(), - text="ping", - sender=None, - recipients=None, - opts=None, - ) - - assert client.messages[0].data.message == "ping" diff --git a/tests/test_bots/test_mixins/test_requests/test_notification.py b/tests/test_bots/test_mixins/test_requests/test_notification.py deleted file mode 100644 index 78b1e444..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_notification.py +++ /dev/null @@ -1,45 +0,0 @@ -import pytest - -from botx import MessagePayload - -pytestmark = pytest.mark.asyncio - - -async def test_filling_with_chat_id_from_credentials(client, message): - await client.bot.send_notification( - credentials=message.credentials, - payload=MessagePayload(text="some text"), - ) - - assert client.notifications[0].result.body == "some text" - - -async def test_filling_with_ids_if_passed(client, message): - await client.bot.send_notification( - message.credentials, - group_chat_ids=[message.user.group_chat_id], - payload=MessagePayload(text="some text"), - ) - - assert client.notifications[0].result.body == "some text" - - -async def test_send_to_all_if_ids_omitted(client, message): - text = "some text" - credentials = message.credentials.copy(update={"chat_id": None}) - - await client.bot.send_notification(credentials, MessagePayload(text=text)) - - assert client.notifications[0].result.body == text - - -async def test_direct_notification_chat_id_required(client, message): - credentials = message.credentials.copy(update={"chat_id": None}) - - with pytest.raises(ValueError) as exc: - await client.bot.send_direct_notification( - credentials, - payload=MessagePayload(text="some text"), - ) - - assert "chat_id is required" in str(exc.value) diff --git a/tests/test_bots/test_mixins/test_requests/test_smartapps.py b/tests/test_bots/test_mixins/test_requests/test_smartapps.py deleted file mode 100644 index 1cde5887..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_smartapps.py +++ /dev/null @@ -1,64 +0,0 @@ -from typing import Any, Dict -from uuid import UUID - -import pytest - -from botx import Message, TestClient -from botx.clients.methods.v3.smartapps.smartapp_event import SmartAppEvent -from botx.clients.methods.v3.smartapps.smartapp_notification import SmartAppNotification -from botx.models.smartapps import SendingSmartAppEvent - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures", "tests.fixtures.smartapps") - - -async def test_smartapp_event( - client: TestClient, - message: Message, - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -): - await client.bot.send_smartapp_event( - credentials=message.credentials, - smartapp_event=SendingSmartAppEvent( - ref=ref, - smartapp_id=smartapp_id, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - data=smartapp_data, - ), - ) - - assert client.requests[0] == SmartAppEvent( - ref=ref, - smartapp_id=smartapp_id, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - data=smartapp_data, - ) - - -async def test_smartapp_notification( - client: TestClient, - message: Message, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_counter: int, -): - await client.bot.send_smartapp_notification( - credentials=message.credentials, - smartapp_notification=SmartAppNotification( - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - smartapp_counter=smartapp_counter, - ), - ) - - assert client.requests[0] == SmartAppNotification( - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - smartapp_counter=smartapp_counter, - ) diff --git a/tests/test_bots/test_mixins/test_requests/test_stickers.py b/tests/test_bots/test_mixins/test_requests/test_stickers.py deleted file mode 100644 index 0b6ba33c..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_stickers.py +++ /dev/null @@ -1,76 +0,0 @@ -from uuid import uuid4 - -import pytest - -from botx.clients.methods.v3.stickers.add_sticker import AddSticker -from botx.clients.methods.v3.stickers.create_sticker_pack import CreateStickerPack -from botx.clients.methods.v3.stickers.delete_sticker import DeleteSticker -from botx.clients.methods.v3.stickers.delete_sticker_pack import DeleteStickerPack -from botx.clients.methods.v3.stickers.edit_sticker_pack import EditStickerPack -from botx.clients.methods.v3.stickers.sticker import GetSticker -from botx.clients.methods.v3.stickers.sticker_pack import GetStickerPack -from botx.clients.methods.v3.stickers.sticker_pack_list import GetStickerPackList -from botx.testing.content import PNG_DATA - -pytestmark = pytest.mark.asyncio - - -async def test_get_sticker_pack_list(client, message): - await client.bot.get_sticker_pack_list(message.credentials) - assert isinstance(client.requests[0], GetStickerPackList) - - -async def test_get_sticker_pack(client, message): - await client.bot.get_sticker_pack(message.credentials, pack_id=uuid4()) - assert isinstance(client.requests[0], GetStickerPack) - - -async def test_get_sticker_from_pack(client, message): - await client.bot.get_sticker_from_pack( - message.credentials, - pack_id=uuid4(), - sticker_id=uuid4(), - ) - assert isinstance(client.requests[0], GetSticker) - - -async def test_create_sticker_pack(client, message): - await client.bot.create_sticker_pack( - message.credentials, - name="Test sticker pack", - user_huid=uuid4(), - ) - assert isinstance(client.requests[0], CreateStickerPack) - - -async def test_add_sticker_into_pack(client, message): - await client.bot.add_sticker( - message.credentials, - pack_id=uuid4(), - emoji="🐢", - image=PNG_DATA, - ) - assert isinstance(client.requests[0], AddSticker) - - -async def test_edit_sticker_pack(client, message): - await client.bot.edit_sticker_pack( - message.credentials, - pack_id=uuid4(), - name="New test sticker pack", - ) - assert isinstance(client.requests[0], EditStickerPack) - - -async def test_delete_sticker_pack(client, message): - await client.bot.delete_sticker_pack(message.credentials, pack_id=uuid4()) - assert isinstance(client.requests[0], DeleteStickerPack) - - -async def test_delete_sticker_from_pack(client, message): - await client.bot.delete_sticker( - message.credentials, - pack_id=uuid4(), - sticker_id=uuid4(), - ) - assert isinstance(client.requests[0], DeleteSticker) diff --git a/tests/test_bots/test_mixins/test_requests/test_users.py b/tests/test_bots/test_mixins/test_requests/test_users.py deleted file mode 100644 index 15a9a58b..00000000 --- a/tests/test_bots/test_mixins/test_requests/test_users.py +++ /dev/null @@ -1,33 +0,0 @@ -import pytest - -from botx.clients.methods.v3.users.by_email import ByEmail -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.clients.methods.v3.users.by_login import ByLogin - -pytestmark = pytest.mark.asyncio - - -async def test_search_requires_one_of_params(client, message): - with pytest.raises(ValueError): - await client.bot.search_user(message.credentials) - - -async def test_search_using_huid_method(client, message): - await client.bot.search_user(message.credentials, user_huid=message.user_huid) - - assert isinstance(client.requests[0], ByHUID) - - -async def test_search_using_email_method(client, message): - await client.bot.search_user(message.credentials, email=message.user.upn) - - assert isinstance(client.requests[0], ByEmail) - - -async def test_search_using_ad_method(client, message): - await client.bot.search_user( - message.credentials, - ad=(message.user.ad_login, message.user.ad_domain), - ) - - assert isinstance(client.requests[0], ByLogin) diff --git a/tests/test_bots/test_mixins/test_sending/test_answer_message.py b/tests/test_bots/test_mixins/test_sending/test_answer_message.py deleted file mode 100644 index d22f39d7..00000000 --- a/tests/test_bots/test_mixins/test_sending/test_answer_message.py +++ /dev/null @@ -1,32 +0,0 @@ -import pytest - -from botx import File - -pytestmark = pytest.mark.asyncio - - -async def test_answer_message_is_notification(bot, client, message): - await bot.answer_message("some text", message) - - message = client.notifications[0] - assert message.result.body == "some text" - - -async def test_answer_message_with_file_is_notification(bot, client, message): - file = File.from_string("some content", "file.txt") - await bot.answer_message( - "some text", - message, - file=file, - ) - - message = client.notifications[0] - assert message.result.body == "some text" - assert message.file == file - - -async def test_answer_message_with_metadata(bot, client, message): - await bot.answer_message("some text", message, metadata={"foo": "bar"}) - - message = client.notifications[0] - assert message.result.metadata == {"foo": "bar"} diff --git a/tests/test_bots/test_mixins/test_sending/test_errors.py b/tests/test_bots/test_mixins/test_sending/test_errors.py deleted file mode 100644 index d8a0b863..00000000 --- a/tests/test_bots/test_mixins/test_sending/test_errors.py +++ /dev/null @@ -1,11 +0,0 @@ -import pytest - -from botx.exceptions import UnknownBotError - -pytestmark = pytest.mark.asyncio - - -async def test_error_for_sending_to_unknown_host(bot, message): - bot.bot_accounts = [] - with pytest.raises(UnknownBotError): - await bot.answer_message("some text", message) diff --git a/tests/test_bots/test_mixins/test_sending/test_send.py b/tests/test_bots/test_mixins/test_sending/test_send.py deleted file mode 100644 index b49bc7eb..00000000 --- a/tests/test_bots/test_mixins/test_sending/test_send.py +++ /dev/null @@ -1,67 +0,0 @@ -import uuid - -import pytest - -from botx import File, SendingMessage - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def sending_file(): - return File.from_string("some content", "file.txt") - - -@pytest.fixture() -def metadata(): - return {"account_id": 94} - - -@pytest.fixture() -def sending_message(message, metadata, sending_file): - sending_message = SendingMessage.from_message( - text="some text", - file=sending_file, - message=message, - ) - sending_message.add_keyboard_button(command="/command", label="keyboard") - sending_message.add_bubble(command="/command", label="bubble") - sending_message.metadata = metadata - return sending_message - - -async def test_using_notification_route(bot, client, sending_message): - await bot.send(sending_message) - - assert client.notifications[0] - - -async def test_sending_notification_using_send(bot, client, sending_message, metadata): - sending_message.credentials.sync_id = None - - await bot.send(sending_message) - - assert len(client.notifications) - notification = client.notifications[0] - - assert notification.result.metadata == metadata - - -async def test_sending_update_using_send(bot, client, sending_message): - sending_message.credentials.message_id = uuid.uuid4() - - await bot.send(sending_message, update=True) - - assert client.message_updates[0].sync_id == sending_message.credentials.message_id - - -async def test_returning_event_id_from_notification(bot, client, sending_message): - sending_message.credentials.sync_id = None - assert await bot.send(sending_message) - - -async def test_setting_custom_id_for_notification(bot, client, sending_message): - message_id = uuid.uuid4() - sending_message.credentials.message_id = message_id - notification_id = await bot.send(sending_message) - assert notification_id == message_id diff --git a/tests/test_bots/test_mixins/test_sending/test_send_file.py b/tests/test_bots/test_mixins/test_sending/test_send_file.py deleted file mode 100644 index d4d22ac9..00000000 --- a/tests/test_bots/test_mixins/test_sending/test_send_file.py +++ /dev/null @@ -1,27 +0,0 @@ -import pytest - -from botx import File - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def sending_file(): - return File.from_string("some content", "file.txt") - - -async def test_send_file_is_notification(bot, client, message, sending_file): - await bot.send_file(sending_file, message.credentials) - - message = client.notifications[0] - assert message.file == sending_file - - -async def test_using_notification(bot, client, message, sending_file): - await bot.send_file( - sending_file, - message.credentials.copy(update={"sync_id": None}), - ) - - message = client.notifications[0] - assert message.file == sending_file diff --git a/tests/test_bots/test_mixins/test_sending/test_send_message.py b/tests/test_bots/test_mixins/test_sending/test_send_message.py deleted file mode 100644 index 34c51cf2..00000000 --- a/tests/test_bots/test_mixins/test_sending/test_send_message.py +++ /dev/null @@ -1,35 +0,0 @@ -import pytest - -from botx import File - -pytestmark = pytest.mark.asyncio - - -async def test_sending_command_result(bot, client, message): - await bot.send_message( - "some text", - message.credentials, - ) - - assert client.notifications[0] - - -async def test_sending_notification_using_send_message(bot, client, message): - await bot.send_message( - "some text", - message.credentials.copy(update={"sync_id": None}), - ) - - assert client.notifications[0] - - -async def test_adding_file(bot, client, message): - sending_file = File.from_string("some content", "file.txt") - await bot.send_message( - "some text", - message.credentials, - file=sending_file.file, - ) - - command_result = client.notifications[0] - assert command_result.file == sending_file diff --git a/tests/test_clients/fixtures.py b/tests/test_clients/fixtures.py deleted file mode 100644 index cd2837d8..00000000 --- a/tests/test_clients/fixtures.py +++ /dev/null @@ -1,11 +0,0 @@ -import pytest - -from botx import AsyncClient, Client - - -@pytest.fixture(params=(AsyncClient, Client)) -def requests_client(request, client): - if issubclass(request.param, AsyncClient): - return client.bot.client - - return client.bot.sync_client diff --git a/tests/test_clients/test_clients/test_async_client/test_execute.py b/tests/test_clients/test_clients/test_async_client/test_execute.py deleted file mode 100644 index cd0a3281..00000000 --- a/tests/test_clients/test_clients/test_async_client/test_execute.py +++ /dev/null @@ -1,51 +0,0 @@ -import uuid - -import pytest -from httpx import ConnectError, Request, Response - -from botx.clients.methods.v2.bots.token import Token -from botx.exceptions import BotXConnectError, BotXJSONDecodeError - -try: - from unittest.mock import AsyncMock -except ImportError: - from unittest.mock import MagicMock - - # Used for compatibility with python 3.7 - class AsyncMock(MagicMock): - async def __call__(self, *args, **kwargs): - return super(AsyncMock, self).__call__(*args, **kwargs) - - -@pytest.fixture() -def token_method(): - return Token(host="example.cts", bot_id=uuid.uuid4(), signature="signature") - - -@pytest.fixture() -def mock_http_client(): - return AsyncMock() - - -@pytest.mark.asyncio() -async def test_raising_connection_error(client, token_method, mock_http_client): - request = Request(token_method.http_method, token_method.url) - mock_http_client.request.side_effect = ConnectError("Test error", request=request) - - client.bot.client.http_client = mock_http_client - botx_request = client.bot.client.build_request(token_method) - - with pytest.raises(BotXConnectError): - await client.bot.client.execute(botx_request) - - -@pytest.mark.asyncio() -async def test_raising_decode_error(client, token_method, mock_http_client): - response = Response(status_code=418, text="Wrong json") - mock_http_client.request.return_value = response - - client.bot.client.http_client = mock_http_client - botx_request = client.bot.client.build_request(token_method) - - with pytest.raises(BotXJSONDecodeError): - await client.bot.client.execute(botx_request) diff --git a/tests/test_clients/test_clients/test_sync_client/test_execute.py b/tests/test_clients/test_clients/test_sync_client/test_execute.py deleted file mode 100644 index 5e90e9ef..00000000 --- a/tests/test_clients/test_clients/test_sync_client/test_execute.py +++ /dev/null @@ -1,46 +0,0 @@ -import uuid -from unittest.mock import Mock - -import pytest -from httpx import ConnectError, Request, Response - -from botx.clients.methods.v2.bots.token import Token -from botx.exceptions import BotXConnectError, BotXJSONDecodeError - - -@pytest.fixture() -def token_method(): - return Token(host="example.cts", bot_id=uuid.uuid4(), signature="signature") - - -@pytest.fixture() -def mock_http_client(): - return Mock() - - -def test_execute_without_explicit_host(client, token_method): - request = client.bot.sync_client.build_request(token_method) - - assert client.bot.sync_client.execute(request) - - -def test_raising_connection_error(client, token_method, mock_http_client): - request = Request(token_method.http_method, token_method.url) - mock_http_client.request.side_effect = ConnectError("Test error", request=request) - - client.bot.sync_client.http_client = mock_http_client - botx_request = client.bot.sync_client.build_request(token_method) - - with pytest.raises(BotXConnectError): - client.bot.sync_client.execute(botx_request) - - -def test_raising_decode_error(client, token_method, mock_http_client): - response = Response(status_code=418, text="Wrong json") - mock_http_client.request.return_value = response - - client.bot.sync_client.http_client = mock_http_client - botx_request = client.bot.sync_client.build_request(token_method) - - with pytest.raises(BotXJSONDecodeError): - client.bot.sync_client.execute(botx_request) diff --git a/tests/test_clients/test_methods/test_base/test_empty_error_handlers.py b/tests/test_clients/test_methods/test_base/test_empty_error_handlers.py deleted file mode 100644 index 5106ff1d..00000000 --- a/tests/test_clients/test_methods/test_base/test_empty_error_handlers.py +++ /dev/null @@ -1,15 +0,0 @@ -from botx.clients.methods.base import BotXMethod - - -class TestMethod(BotXMethod): - __test__ = False - - __url__ = "/path/to/example" - __method__ = "GET" - __returning__ = str - - -def test_method_empty_error_handlers(): - test_method = TestMethod() - - assert test_method.__errors_handlers__ == {} diff --git a/tests/test_clients/test_methods/test_base/test_unhandled_error.py b/tests/test_clients/test_methods/test_base/test_unhandled_error.py deleted file mode 100644 index 8f22e8ee..00000000 --- a/tests/test_clients/test_methods/test_base/test_unhandled_error.py +++ /dev/null @@ -1,66 +0,0 @@ -import uuid - -import pytest - -from botx import BotXAPIError, ChatTypes -from botx.clients.methods.errors.bot_is_not_admin import BotIsNotAdminData -from botx.clients.methods.v2.bots.token import Token -from botx.clients.methods.v3.chats.create import Create -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - -IM_A_TEAPOT = 418 # This status code added in python 3.9 - - -async def test_raising_base_api_error_if_empty_handlers(client, requests_client): - method = Token(host="example.com", bot_id=uuid.uuid4(), signature="signature") - - errors_to_raise = { - Token: ( - IM_A_TEAPOT, - BotIsNotAdminData(sender=uuid.uuid4(), group_chat_id=uuid.uuid4()), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(BotXAPIError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - -async def test_raising_base_api_error_if_unhandled(client, requests_client): - method = Create( - host="example.com", - name="test name", - members=[uuid.uuid4()], - chat_type=ChatTypes.group_chat, - shared_history=False, - ) - - method.__errors_handlers__[IM_A_TEAPOT] = [] - - errors_to_raise = { - Create: ( - IM_A_TEAPOT, - BotIsNotAdminData(sender=uuid.uuid4(), group_chat_id=uuid.uuid4()), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(BotXAPIError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/__init__.py b/tests/test_clients/test_methods/test_errors/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_errors/test_bot_is_not_admin.py b/tests/test_clients/test_methods/test_errors/test_bot_is_not_admin.py deleted file mode 100644 index 569296e4..00000000 --- a/tests/test_clients/test_methods/test_errors/test_bot_is_not_admin.py +++ /dev/null @@ -1,39 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.bot_is_not_admin import ( - BotIsNotAdminData, - BotIsNotAdminError, -) -from botx.clients.methods.v3.chats.add_user import AddUser -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_bot_is_not_admin(client, requests_client): - method = AddUser( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4() for _ in range(10)], - ) - errors_to_raise = { - AddUser: ( - HTTPStatus.FORBIDDEN, - BotIsNotAdminData(sender=uuid.uuid4(), group_chat_id=method.group_chat_id), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(BotIsNotAdminError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_bot_not_found.py b/tests/test_clients/test_methods/test_errors/test_bot_not_found.py deleted file mode 100644 index 1ae420f1..00000000 --- a/tests/test_clients/test_methods/test_errors/test_bot_not_found.py +++ /dev/null @@ -1,28 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.bot_not_found import BotNotFoundError -from botx.clients.methods.v2.bots.token import Token -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_bot_not_found_error(client, requests_client): - method = Token(host="example.com", bot_id=uuid.uuid4(), signature="signature") - - errors_to_raise = {Token: (HTTPStatus.NOT_FOUND, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(BotNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_chat_creation_disallowed.py b/tests/test_clients/test_methods/test_errors/test_chat_creation_disallowed.py deleted file mode 100644 index 4ffe31cf..00000000 --- a/tests/test_clients/test_methods/test_errors/test_chat_creation_disallowed.py +++ /dev/null @@ -1,43 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx import ChatTypes -from botx.clients.methods.errors.chat_creation_disallowed import ( - ChatCreationDisallowedData, - ChatCreationDisallowedError, -) -from botx.clients.methods.v3.chats.create import Create -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_chat_creation_disallowed(client, requests_client): - method = Create( - host="example.com", - name="test name", - members=[uuid.uuid4()], - chat_type=ChatTypes.group_chat, - shared_history=False, - ) - - errors_to_raise = { - Create: ( - HTTPStatus.FORBIDDEN, - ChatCreationDisallowedData(bot_id=uuid.uuid4()), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(ChatCreationDisallowedError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_chat_creation_error.py b/tests/test_clients/test_methods/test_errors/test_chat_creation_error.py deleted file mode 100644 index 1d238028..00000000 --- a/tests/test_clients/test_methods/test_errors/test_chat_creation_error.py +++ /dev/null @@ -1,35 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx import ChatTypes -from botx.clients.methods.errors.chat_creation_error import ChatCreationError -from botx.clients.methods.v3.chats.create import Create -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_chat_creation_error(client, requests_client): - method = Create( - host="example.com", - name="test name", - members=[uuid.uuid4()], - chat_type=ChatTypes.group_chat, - shared_history=False, - ) - - errors_to_raise = {Create: (HTTPStatus.UNPROCESSABLE_ENTITY, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(ChatCreationError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_chat_is_not_modifiable.py b/tests/test_clients/test_methods/test_errors/test_chat_is_not_modifiable.py deleted file mode 100644 index 3904fbda..00000000 --- a/tests/test_clients/test_methods/test_errors/test_chat_is_not_modifiable.py +++ /dev/null @@ -1,40 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.chat_is_not_modifiable import ( - PersonalChatIsNotModifiableData, - PersonalChatIsNotModifiableError, -) -from botx.clients.methods.v3.chats.add_user import AddUser -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_chat_is_not_modifiable(client, requests_client): - method = AddUser( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4()], - ) - - errors_to_raise = { - AddUser: ( - HTTPStatus.FORBIDDEN, - PersonalChatIsNotModifiableData(group_chat_id=method.group_chat_id), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(PersonalChatIsNotModifiableError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_chat_not_found.py b/tests/test_clients/test_methods/test_errors/test_chat_not_found.py deleted file mode 100644 index 790d258c..00000000 --- a/tests/test_clients/test_methods/test_errors/test_chat_not_found.py +++ /dev/null @@ -1,40 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.chat_not_found import ( - ChatNotFoundData, - ChatNotFoundError, -) -from botx.clients.methods.v3.chats.add_user import AddUser -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_chat_not_found(client, requests_client): - method = AddUser( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4()], - ) - - errors_to_raise = { - AddUser: ( - HTTPStatus.NOT_FOUND, - ChatNotFoundData(group_chat_id=method.group_chat_id), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(ChatNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_files/__init__.py b/tests/test_clients/test_methods/test_errors/test_files/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_errors/test_files/test_chat_not_found.py b/tests/test_clients/test_methods/test_errors/test_files/test_chat_not_found.py deleted file mode 100644 index 3f691fb0..00000000 --- a/tests/test_clients/test_methods/test_errors/test_files/test_chat_not_found.py +++ /dev/null @@ -1,44 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.files.chat_not_found import ( - ChatNotFoundData, - ChatNotFoundError, -) -from botx.clients.methods.v3.files.download import DownloadFile -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_chat_not_found(client, requests_client): - method = DownloadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file_id=uuid.uuid4(), - is_preview=False, - ) - - errors_to_raise = { - DownloadFile: ( - HTTPStatus.NOT_FOUND, - ChatNotFoundData( - group_chat_id=method.group_chat_id, - error_description="test", - ), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(ChatNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_files/test_file_deleted.py b/tests/test_clients/test_methods/test_errors/test_files/test_file_deleted.py deleted file mode 100644 index 4cdeeb10..00000000 --- a/tests/test_clients/test_methods/test_errors/test_files/test_file_deleted.py +++ /dev/null @@ -1,41 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.files.file_deleted import ( - FileDeletedError, - FileDeletedErrorData, -) -from botx.clients.methods.v3.files.download import DownloadFile -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_file_deleted(client, requests_client): - method = DownloadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file_id=uuid.uuid4(), - is_preview=False, - ) - - errors_to_raise = { - DownloadFile: ( - HTTPStatus.NO_CONTENT, - FileDeletedErrorData(link="/path/to/file", error_description="test"), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(FileDeletedError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_files/test_metadata_not_found.py b/tests/test_clients/test_methods/test_errors/test_files/test_metadata_not_found.py deleted file mode 100644 index 6899c51b..00000000 --- a/tests/test_clients/test_methods/test_errors/test_files/test_metadata_not_found.py +++ /dev/null @@ -1,45 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.files.metadata_not_found import ( - MetadataNotFoundData, - MetadataNotFoundError, -) -from botx.clients.methods.v3.files.download import DownloadFile -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_metadata_found(client, requests_client): - method = DownloadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file_id=uuid.uuid4(), - is_preview=False, - ) - - errors_to_raise = { - DownloadFile: ( - HTTPStatus.NOT_FOUND, - MetadataNotFoundData( - file_id=method.file_id, - group_chat_id=method.group_chat_id, - error_description="test", - ), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(MetadataNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_files/test_without_preview.py b/tests/test_clients/test_methods/test_errors/test_files/test_without_preview.py deleted file mode 100644 index 5d583c12..00000000 --- a/tests/test_clients/test_methods/test_errors/test_files/test_without_preview.py +++ /dev/null @@ -1,45 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.files.without_preview import ( - WithoutPreviewData, - WithoutPreviewError, -) -from botx.clients.methods.v3.files.download import DownloadFile -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_without_preview(client, requests_client): - method = DownloadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file_id=uuid.uuid4(), - is_preview=True, - ) - - errors_to_raise = { - DownloadFile: ( - HTTPStatus.BAD_REQUEST, - WithoutPreviewData( - file_id=method.file_id, - group_chat_id=method.group_chat_id, - error_description="test", - ), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(WithoutPreviewError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_messaging.py b/tests/test_clients/test_methods/test_errors/test_messaging.py deleted file mode 100644 index bda8c716..00000000 --- a/tests/test_clients/test_methods/test_errors/test_messaging.py +++ /dev/null @@ -1,28 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.messaging import MessagingError -from botx.clients.methods.v3.chats.info import Info -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_messaging_error(client, requests_client): - method = Info(host="example.com", group_chat_id=uuid.uuid4()) - - errors_to_raise = {Info: (HTTPStatus.BAD_REQUEST, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(MessagingError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_permissions.py b/tests/test_clients/test_methods/test_errors/test_permissions.py deleted file mode 100644 index c2b77fcd..00000000 --- a/tests/test_clients/test_methods/test_errors/test_permissions.py +++ /dev/null @@ -1,40 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.permissions import ( - NoPermissionError, - NoPermissionErrorData, -) -from botx.clients.methods.v3.chats.pin_message import PinMessage -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_no_permission(client, requests_client): - method = PinMessage( - host="example.com", - chat_id=uuid.uuid4(), - sync_id=uuid.uuid4(), - ) - - errors_to_raise = { - PinMessage: ( - HTTPStatus.FORBIDDEN, - NoPermissionErrorData(group_chat_id=method.chat_id), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(NoPermissionError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_stickers/__init__.py b/tests/test_clients/test_methods/test_errors/test_stickers/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_errors/test_stickers/test_image_not_valid.py b/tests/test_clients/test_methods/test_errors/test_stickers/test_image_not_valid.py deleted file mode 100644 index 8b5da31f..00000000 --- a/tests/test_clients/test_methods/test_errors/test_stickers/test_image_not_valid.py +++ /dev/null @@ -1,39 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.stickers.image_not_valid import ImageNotValidError -from botx.clients.methods.v3.stickers.add_sticker import AddSticker -from botx.concurrency import callable_to_coroutine -from botx.testing.content import PNG_DATA - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_image_not_valid(client, requests_client): - method = AddSticker( - host="example.com", - pack_id=uuid.uuid4(), - emoji="🐢", - image=PNG_DATA, - ) - - errors_to_raise = { - AddSticker: ( - HTTPStatus.BAD_REQUEST, - {}, - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(ImageNotValidError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_not_found.py b/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_not_found.py deleted file mode 100644 index a781fd48..00000000 --- a/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_not_found.py +++ /dev/null @@ -1,43 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.stickers.sticker_pack_or_sticker_not_found import ( - StickerPackOrStickerNotFoundData, - StickerPackOrStickerNotFoundError, -) -from botx.clients.methods.v3.stickers.sticker import GetSticker -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_sticker_not_found(client, requests_client): - method = GetSticker( - host="example.com", - pack_id=uuid.uuid4(), - sticker_id=uuid.uuid4(), - ) - - errors_to_raise = { - GetSticker: ( - HTTPStatus.NOT_FOUND, - StickerPackOrStickerNotFoundData( - pack_id=method.pack_id, - sticker_id=method.sticker_id, - ), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(StickerPackOrStickerNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_pack_not_found.py b/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_pack_not_found.py deleted file mode 100644 index 942fb839..00000000 --- a/tests/test_clients/test_methods/test_errors/test_stickers/test_sticker_pack_not_found.py +++ /dev/null @@ -1,39 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.stickers.sticker_pack_not_found import ( - StickerPackNotFoundData, - StickerPackNotFoundError, -) -from botx.clients.methods.v3.stickers.sticker_pack import GetStickerPack -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_sticker_pack_not_found(client, requests_client): - method = GetStickerPack( - host="example.com", - pack_id=uuid.uuid4(), - ) - - errors_to_raise = { - GetStickerPack: ( - HTTPStatus.NOT_FOUND, - StickerPackNotFoundData(pack_id=method.pack_id), - ), - } - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(StickerPackNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_unauthorized_bot.py b/tests/test_clients/test_methods/test_errors/test_unauthorized_bot.py deleted file mode 100644 index 27105221..00000000 --- a/tests/test_clients/test_methods/test_errors/test_unauthorized_bot.py +++ /dev/null @@ -1,28 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.unauthorized_bot import InvalidBotCredentials -from botx.clients.methods.v2.bots.token import Token -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_unauthorized_bot_error(client, requests_client): - method = Token(host="example.com", bot_id=uuid.uuid4(), signature="signature") - - errors_to_raise = {Token: (HTTPStatus.UNAUTHORIZED, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(InvalidBotCredentials): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_errors/test_user_not_found.py b/tests/test_clients/test_methods/test_errors/test_user_not_found.py deleted file mode 100644 index 41f36d00..00000000 --- a/tests/test_clients/test_methods/test_errors/test_user_not_found.py +++ /dev/null @@ -1,28 +0,0 @@ -import uuid -from http import HTTPStatus - -import pytest - -from botx.clients.methods.errors.user_not_found import UserNotFoundError -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_user_not_found(client, requests_client): - method = ByHUID(host="example.com", user_huid=uuid.uuid4()) - - errors_to_raise = {ByHUID: (HTTPStatus.NOT_FOUND, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(UserNotFoundError): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_clients/test_methods/test_v2/__init__.py b/tests/test_clients/test_methods/test_v2/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v2/test_bots/__init__.py b/tests/test_clients/test_methods/test_v2/test_bots/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v2/test_bots/test_token.py b/tests/test_clients/test_methods/test_v2/test_bots/test_token.py deleted file mode 100644 index e1bea697..00000000 --- a/tests/test_clients/test_methods/test_v2/test_bots/test_token.py +++ /dev/null @@ -1,19 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v2.bots.token import Token -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_obtaining_token(client, requests_client): - method = Token(host="example.com", bot_id=uuid.uuid4(), signature="signature") - - request = requests_client.build_request(method) - await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].bot_id == method.bot_id - assert client.requests[0].signature == method.signature diff --git a/tests/test_clients/test_methods/test_v3/__init__.py b/tests/test_clients/test_methods/test_v3/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_chats/__init__.py b/tests/test_clients/test_methods/test_v3/test_chats/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_add_admin_role.py b/tests/test_clients/test_methods/test_v3/test_chats/test_add_admin_role.py deleted file mode 100644 index 6d39273b..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_add_admin_role.py +++ /dev/null @@ -1,22 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.add_admin_role import AddAdminRole -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_adding_users(client, requests_client): - method = AddAdminRole( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4() for _ in range(10)], - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].group_chat_id == method.group_chat_id - assert client.requests[0].user_huids == method.user_huids diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_add_user.py b/tests/test_clients/test_methods/test_v3/test_chats/test_add_user.py deleted file mode 100644 index 102083b1..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_add_user.py +++ /dev/null @@ -1,22 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.add_user import AddUser -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_adding_users(client, requests_client): - method = AddUser( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4() for _ in range(10)], - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].group_chat_id == method.group_chat_id - assert client.requests[0].user_huids == method.user_huids diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_chat_list.py b/tests/test_clients/test_methods/test_v3/test_chats/test_chat_list.py deleted file mode 100644 index 5e706bd4..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_chat_list.py +++ /dev/null @@ -1,24 +0,0 @@ -import pytest - -from botx.clients.methods.v3.chats.chat_list import ChatList -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_retrieving_bot_chats(client, requests_client): - method = ChatList(host="example.com") - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - bot_chats = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert isinstance(list(bot_chats), list) - - assert len(bot_chats) == 1 - - assert len(bot_chats[0].members) == 1 diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_create.py b/tests/test_clients/test_methods/test_v3/test_chats/test_create.py deleted file mode 100644 index 0328cb1d..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_create.py +++ /dev/null @@ -1,25 +0,0 @@ -import uuid - -import pytest - -from botx import ChatTypes -from botx.clients.methods.v3.chats.create import Create -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_chat_creation(client, requests_client): - method = Create( - host="example.com", - name="test name", - members=[uuid.uuid4()], - chat_type=ChatTypes.group_chat, - shared_history=False, - ) - - request = requests_client.build_request(method) - await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].name == method.name - assert client.requests[0].members == method.members diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_info.py b/tests/test_clients/test_methods/test_v3/test_chats/test_info.py deleted file mode 100644 index 4c9a88d4..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_info.py +++ /dev/null @@ -1,24 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.info import Info -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_retrieving_info(client, requests_client): - method = Info(host="example.com", group_chat_id=uuid.uuid4()) - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - info = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert info.members - - assert client.requests[0].group_chat_id == method.group_chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_pin_message.py b/tests/test_clients/test_methods/test_v3/test_chats/test_pin_message.py deleted file mode 100644 index 8b634e59..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_pin_message.py +++ /dev/null @@ -1,21 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.pin_message import PinMessage -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_pinning_message(client, requests_client): - chat_id = uuid.uuid4() - sync_id = uuid.uuid4() - method = PinMessage(host="example.com", chat_id=chat_id, sync_id=sync_id) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].chat_id == method.chat_id - assert client.requests[0].sync_id == method.sync_id diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_remove_user.py b/tests/test_clients/test_methods/test_v3/test_chats/test_remove_user.py deleted file mode 100644 index f87ef12a..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_remove_user.py +++ /dev/null @@ -1,21 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.remove_user import RemoveUser -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio - - -async def test_removing_users(client, requests_client): - method = RemoveUser( - host="example.com", - group_chat_id=uuid.uuid4(), - user_huids=[uuid.uuid4() for _ in range(10)], - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].group_chat_id == method.group_chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_disable.py b/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_disable.py deleted file mode 100644 index f50b0307..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_disable.py +++ /dev/null @@ -1,18 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.stealth_disable import StealthDisable -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_disabling_stealth(client, requests_client): - method = StealthDisable(host="example.com", group_chat_id=uuid.uuid4()) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].group_chat_id == method.group_chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_set.py b/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_set.py deleted file mode 100644 index 980e5bf5..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_stealth_set.py +++ /dev/null @@ -1,18 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.stealth_set import StealthSet -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_enabling_stealth(client, requests_client): - method = StealthSet(host="example.com", group_chat_id=uuid.uuid4()) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].group_chat_id == method.group_chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_chats/test_unpin_message.py b/tests/test_clients/test_methods/test_v3/test_chats/test_unpin_message.py deleted file mode 100644 index 8d192f29..00000000 --- a/tests/test_clients/test_methods/test_v3/test_chats/test_unpin_message.py +++ /dev/null @@ -1,19 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.chats.unpin_message import UnpinMessage -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_unpinning_message(client, requests_client): - chat_id = uuid.uuid4() - method = UnpinMessage(host="example.com", chat_id=chat_id) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].chat_id == method.chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_command/__init__.py b/tests/test_clients/test_methods/test_v3/test_command/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_command/test_command_result.py b/tests/test_clients/test_methods/test_v3/test_command/test_command_result.py deleted file mode 100644 index 8eb42a59..00000000 --- a/tests/test_clients/test_methods/test_v3/test_command/test_command_result.py +++ /dev/null @@ -1,24 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.command.command_result import CommandResult -from botx.clients.types.message_payload import ResultPayload -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_sending_command_result(client, requests_client): - method = CommandResult( - host="example.com", - sync_id=uuid.uuid4(), - bot_id=uuid.uuid4(), - result=ResultPayload(body="test"), - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].result.body == method.result.body diff --git a/tests/test_clients/test_methods/test_v3/test_events/__init__.py b/tests/test_clients/test_methods/test_v3/test_events/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_events/test_edit_event.py b/tests/test_clients/test_methods/test_v3/test_events/test_edit_event.py deleted file mode 100644 index 41265103..00000000 --- a/tests/test_clients/test_methods/test_v3/test_events/test_edit_event.py +++ /dev/null @@ -1,23 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.events.edit_event import EditEvent -from botx.clients.types.message_payload import UpdatePayload -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_sending_edit_event(client, requests_client): - method = EditEvent( - host="example.com", - sync_id=uuid.uuid4(), - result=UpdatePayload(body="test"), - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].result.body == method.result.body diff --git a/tests/test_clients/test_methods/test_v3/test_files/__init__.py b/tests/test_clients/test_methods/test_v3/test_files/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_files/test_download.py b/tests/test_clients/test_methods/test_v3/test_files/test_download.py deleted file mode 100644 index 2eedce1a..00000000 --- a/tests/test_clients/test_methods/test_v3/test_files/test_download.py +++ /dev/null @@ -1,32 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.files.download import DownloadFile -from botx.concurrency import callable_to_coroutine -from botx.models.files import File - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_download_file(client, requests_client): - file_id = uuid.uuid4() - method = DownloadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file_id=file_id, - is_preview=False, - ) - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - file = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert isinstance(file, File) - - assert client.requests[0].file_id == file_id diff --git a/tests/test_clients/test_methods/test_v3/test_files/test_upload.py b/tests/test_clients/test_methods/test_v3/test_files/test_upload.py deleted file mode 100644 index d1c1da45..00000000 --- a/tests/test_clients/test_methods/test_v3/test_files/test_upload.py +++ /dev/null @@ -1,35 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.files.upload import UploadFile -from botx.concurrency import callable_to_coroutine -from botx.models.files import File -from botx.testing.content import PNG_DATA - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_upload_file(client, requests_client): - file_name = "image.png" - - image = File(file_name=file_name, data=PNG_DATA) - method = UploadFile( - host="example.com", - group_chat_id=uuid.uuid4(), - file=image, - meta={}, - ) - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - meta_file = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert meta_file.file_name == file_name - - assert client.requests[0].file.file_name == file_name diff --git a/tests/test_clients/test_methods/test_v3/test_notification/__init__.py b/tests/test_clients/test_methods/test_v3/test_notification/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_notification/test_notification.py b/tests/test_clients/test_methods/test_v3/test_notification/test_notification.py deleted file mode 100644 index f0791863..00000000 --- a/tests/test_clients/test_methods/test_v3/test_notification/test_notification.py +++ /dev/null @@ -1,24 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.notification.notification import Notification -from botx.clients.types.message_payload import ResultPayload -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_sending_notification(client, requests_client): - method = Notification( - host="example.com", - group_chat_ids=[uuid.uuid4()], - bot_id=uuid.uuid4(), - result=ResultPayload(body="test"), - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].result.body == method.result.body diff --git a/tests/test_clients/test_methods/test_v3/test_notification/test_notification_direct.py b/tests/test_clients/test_methods/test_v3/test_notification/test_notification_direct.py deleted file mode 100644 index b67a6fff..00000000 --- a/tests/test_clients/test_methods/test_v3/test_notification/test_notification_direct.py +++ /dev/null @@ -1,24 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.notification.direct_notification import NotificationDirect -from botx.clients.types.message_payload import ResultPayload -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_sending_direct_notification(client, requests_client): - method = NotificationDirect( - host="example.com", - group_chat_id=uuid.uuid4(), - bot_id=uuid.uuid4(), - result=ResultPayload(body="test"), - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].result.body == method.result.body diff --git a/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_event.py b/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_event.py deleted file mode 100644 index decc32ee..00000000 --- a/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_event.py +++ /dev/null @@ -1,42 +0,0 @@ -from typing import Any, Dict, Union -from uuid import UUID - -import pytest - -from botx import AsyncClient, Client, TestClient -from botx.clients.methods.v3.smartapps.smartapp_event import SmartAppEvent -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures", "tests.fixtures.smartapps") - - -async def test_smartapp_event( - client: TestClient, - requests_client: Union[AsyncClient, Client], - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -) -> None: - method = SmartAppEvent( - host="example.com", # type: ignore [call-arg] - ref=ref, - smartapp_id=smartapp_id, - data=smartapp_data, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert isinstance(client.requests[0], SmartAppEvent) - smartapp_event = client.requests[0] - - assert smartapp_event.ref == ref - assert smartapp_event.smartapp_id == smartapp_id - assert smartapp_event.smartapp_api_version == smartapp_api_version - assert smartapp_event.group_chat_id == group_chat_id - assert smartapp_event.data == smartapp_data diff --git a/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_notification.py b/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_notification.py deleted file mode 100644 index 33ec0d25..00000000 --- a/tests/test_clients/test_methods/test_v3/test_smartapps/test_smartapp_notification.py +++ /dev/null @@ -1,36 +0,0 @@ -from typing import Union -from uuid import UUID - -import pytest - -from botx import AsyncClient, Client, TestClient -from botx.clients.methods.v3.smartapps.smartapp_notification import SmartAppNotification -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures", "tests.fixtures.smartapps") - - -async def test_smartapp_event( - client: TestClient, - requests_client: Union[AsyncClient, Client], - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_counter: int, -) -> None: - method = SmartAppNotification( - host="example.com", # type: ignore [call-arg] - group_chat_id=group_chat_id, - smartapp_counter=smartapp_counter, - smartapp_api_version=smartapp_api_version, - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert isinstance(client.requests[0], SmartAppNotification) - client.requests[0] - - assert client.requests[0].smartapp_counter == smartapp_counter - assert client.requests[0].smartapp_api_version == smartapp_api_version - assert client.requests[0].group_chat_id == group_chat_id diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/__init__.py b/tests/test_clients/test_methods/test_v3/test_stickers/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_add_sticker.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_add_sticker.py deleted file mode 100644 index c485bcdf..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_add_sticker.py +++ /dev/null @@ -1,31 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.add_sticker import AddSticker -from botx.concurrency import callable_to_coroutine -from botx.testing.content import PNG_DATA - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_add_sticker_into_sticker_pack(client, requests_client): - emoji = "🐢" - - method = AddSticker( - pack_id=uuid.uuid4(), - emoji=emoji, - image=PNG_DATA, - host="example.com", - ) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker.emoji == emoji diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_create_sticker_pack.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_create_sticker_pack.py deleted file mode 100644 index ea8702b2..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_create_sticker_pack.py +++ /dev/null @@ -1,29 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.create_sticker_pack import CreateStickerPack -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_create_sticker_pack(client, requests_client): - sticker_pack_name = "Test sticker pack" - - method = CreateStickerPack( - name=sticker_pack_name, - host="example.com", - user_huid=uuid.uuid4(), - ) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker_pack = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker_pack.name == sticker_pack_name diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker.py deleted file mode 100644 index c80aa0bb..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker.py +++ /dev/null @@ -1,28 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.delete_sticker import DeleteSticker -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_delete_sticker(client, requests_client): - - method = DeleteSticker( - pack_id=uuid.uuid4(), - sticker_id=uuid.uuid4(), - host="example.com", - ) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - result = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert result == "sticker_deleted" diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker_pack.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker_pack.py deleted file mode 100644 index 6cb89d10..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_delete_sticker_pack.py +++ /dev/null @@ -1,24 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.delete_sticker_pack import DeleteStickerPack -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_delete_sticker_pack(client, requests_client): - - method = DeleteStickerPack(pack_id=uuid.uuid4(), host="example.com") - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - result = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert result == "sticker_pack_deleted" diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_edit_sticker_pack.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_edit_sticker_pack.py deleted file mode 100644 index af5dc62d..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_edit_sticker_pack.py +++ /dev/null @@ -1,29 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.edit_sticker_pack import EditStickerPack -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_edit_sticker_pack(client, requests_client): - sticker_pack_name = "Test sticker pack" - - method = EditStickerPack( - pack_id=uuid.uuid4(), - name=sticker_pack_name, - host="example.com", - ) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker_pack = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker_pack.name == sticker_pack_name diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_from_sticker_pack.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_from_sticker_pack.py deleted file mode 100644 index a21003a9..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_from_sticker_pack.py +++ /dev/null @@ -1,29 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.sticker import GetSticker -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_get_sticker_from_sticker_pack(client, requests_client): - emoji = "🐢" - - method = GetSticker( - pack_id=uuid.uuid4(), - sticker_id=uuid.uuid4(), - host="example.com", - ) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker.emoji == emoji diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack.py deleted file mode 100644 index ef36c34c..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack.py +++ /dev/null @@ -1,25 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.stickers.sticker_pack import GetStickerPack -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_get_sticker_pack(client, requests_client): - sticker_pack_name = "Test sticker pack" - - method = GetStickerPack(pack_id=uuid.uuid4(), host="example.com") - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker_pack = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker_pack.name == sticker_pack_name diff --git a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack_list.py b/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack_list.py deleted file mode 100644 index 1f6be276..00000000 --- a/tests/test_clients/test_methods/test_v3/test_stickers/test_get_sticker_pack_list.py +++ /dev/null @@ -1,23 +0,0 @@ -import pytest - -from botx.clients.methods.v3.stickers.sticker_pack_list import GetStickerPackList -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_get_sticker_pack_list(client, requests_client): - sticker_pack_name = "Test sticker pack" - - method = GetStickerPackList(host="example.com", limit=1) - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - sticker_pack_list = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert sticker_pack_list.packs[0].name == sticker_pack_name diff --git a/tests/test_clients/test_methods/test_v3/test_users/__init__.py b/tests/test_clients/test_methods/test_v3/test_users/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v3/test_users/test_by_email.py b/tests/test_clients/test_methods/test_v3/test_users/test_by_email.py deleted file mode 100644 index 2504b01c..00000000 --- a/tests/test_clients/test_methods/test_v3/test_users/test_by_email.py +++ /dev/null @@ -1,23 +0,0 @@ -import pytest - -from botx.clients.methods.v3.users.by_email import ByEmail -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_search_by_huid(client, requests_client): - method = ByEmail(host="example.com", email="test@example.com") - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - user = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert user.emails == [method.email] - - assert client.requests[0].email == method.email diff --git a/tests/test_clients/test_methods/test_v3/test_users/test_by_huid.py b/tests/test_clients/test_methods/test_v3/test_users/test_by_huid.py deleted file mode 100644 index 842cbc0d..00000000 --- a/tests/test_clients/test_methods/test_v3/test_users/test_by_huid.py +++ /dev/null @@ -1,25 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v3.users.by_huid import ByHUID -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_search_by_huid(client, requests_client): - method = ByHUID(host="example.com", user_huid=uuid.uuid4()) - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - user = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert user.user_huid == method.user_huid - - assert client.requests[0].user_huid == method.user_huid diff --git a/tests/test_clients/test_methods/test_v3/test_users/test_by_login.py b/tests/test_clients/test_methods/test_v3/test_users/test_by_login.py deleted file mode 100644 index 096fadfc..00000000 --- a/tests/test_clients/test_methods/test_v3/test_users/test_by_login.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from botx.clients.methods.v3.users.by_login import ByLogin -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_search_by_huid(client, requests_client): - method = ByLogin(host="example.com", ad_login="test", ad_domain="example.com") - - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - user = await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) - - assert user.ad_login == method.ad_login - assert user.ad_domain == method.ad_domain - - assert client.requests[0].ad_login == method.ad_login - assert client.requests[0].ad_domain == method.ad_domain diff --git a/tests/test_clients/test_methods/test_v4/__init__.py b/tests/test_clients/test_methods/test_v4/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v4/test_notifications/__init__.py b/tests/test_clients/test_methods/test_v4/test_notifications/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_methods/test_v4/test_notifications/test_internal_bot_notification.py b/tests/test_clients/test_methods/test_v4/test_notifications/test_internal_bot_notification.py deleted file mode 100644 index be7b01ac..00000000 --- a/tests/test_clients/test_methods/test_v4/test_notifications/test_internal_bot_notification.py +++ /dev/null @@ -1,26 +0,0 @@ -import uuid - -import pytest - -from botx.clients.methods.v4.notifications.internal_bot_notification import ( - InternalBotNotification, -) -from botx.clients.types.message_payload import InternalBotNotificationPayload -from botx.concurrency import callable_to_coroutine - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_sending_internal_bot_notification(client, requests_client): - method = InternalBotNotification( - host="example.com", - group_chat_id=uuid.uuid4(), - bot_id=uuid.uuid4(), - data=InternalBotNotificationPayload(message="test"), - ) - - request = requests_client.build_request(method) - assert await callable_to_coroutine(requests_client.execute, request) - - assert client.requests[0].data.message == "test" diff --git a/tests/test_clients/test_types/__init__.py b/tests/test_clients/test_types/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_clients/test_types/test_http.py b/tests/test_clients/test_types/test_http.py deleted file mode 100644 index d3f1fd69..00000000 --- a/tests/test_clients/test_types/test_http.py +++ /dev/null @@ -1,14 +0,0 @@ -import pytest -from pydantic import ValidationError - -from botx.clients.types.http import HTTPResponse - - -def test_response_validation(): - with pytest.raises(ValidationError): - HTTPResponse( - headers={}, - status_code=200, - json_body={"status": "ok"}, - raw_data=b"content", - ) diff --git a/tests/test_collecting/__init__.py b/tests/test_collecting/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_collecting/fixtures.py b/tests/test_collecting/fixtures.py deleted file mode 100644 index ec71704c..00000000 --- a/tests/test_collecting/fixtures.py +++ /dev/null @@ -1,69 +0,0 @@ -import pytest - -from botx import Bot, Collector -from botx.collecting.handlers.handler import Handler - - -class HandlerClass: - def handler_method_snake_case(self) -> None: - """Handler with name in snake case.""" - - def handlerMethodCamelCase(self) -> None: # noqa: N802 - """Handler with name in camel case.""" - - def HandlerMethodPascalCase(self) -> None: # noqa: N802 - """Handler with name in pascal case.""" - - def __call__(self) -> None: - """Handler that is callable class.""" - - -@pytest.fixture() -def handler_as_function(build_handler_for_collector): - return build_handler_for_collector("handler_function") - - -@pytest.fixture() -def handler_as_class(): - return HandlerClass - - -@pytest.fixture() -def handler_as_callable_object(): - return HandlerClass() - - -@pytest.fixture() -def handler_as_normal_method(): - return HandlerClass().handler_method_snake_case - - -@pytest.fixture() -def handler_as_pascal_case_method(): - return HandlerClass().HandlerMethodPascalCase - - -@pytest.fixture() -def handler_as_camel_case_method(): - return HandlerClass().handlerMethodCamelCase - - -@pytest.fixture() -def default_handler(handler_as_function): - return Handler(handler=handler_as_function, body="/default-handler") - - -@pytest.fixture() -def extract_collector(): - def factory(collector_instance): - if isinstance(collector_instance, Bot): - return collector_instance.collector - - return collector_instance - - return factory - - -@pytest.fixture(params=(Collector, Bot)) -def collector_cls(request): - return request.param diff --git a/tests/test_collecting/test_collector/__init__.py b/tests/test_collecting/test_collector/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_collecting/test_collector/test_bodies_generating.py b/tests/test_collecting/test_collector/test_bodies_generating.py deleted file mode 100644 index f534f715..00000000 --- a/tests/test_collecting/test_collector/test_bodies_generating.py +++ /dev/null @@ -1,24 +0,0 @@ -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_generating_body_from_snake_case(handler_as_normal_method): - collector = Collector() - collector.add_handler(handler=handler_as_normal_method) - handler = collector.handler_for("handler_method_snake_case") - assert handler.body == "/handler-method-snake-case" - - -def test_generating_body_from_pascal_case(handler_as_pascal_case_method): - collector = Collector() - collector.add_handler(handler=handler_as_pascal_case_method) - handler = collector.handler_for("HandlerMethodPascalCase") - assert handler.body == "/handler-method-pascal-case" - - -def test_generating_body_from_camel_case(handler_as_camel_case_method): - collector = Collector() - collector.add_handler(handler=handler_as_camel_case_method) - handler = collector.handler_for("handlerMethodCamelCase") - assert handler.body == "/handler-method-camel-case" diff --git a/tests/test_collecting/test_collector/test_collectors_merging/__init__.py b/tests/test_collecting/test_collector/test_collectors_merging/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_collecting/test_collector/test_collectors_merging/test_default_handler_adding.py b/tests/test_collecting/test_collector/test_collectors_merging/test_default_handler_adding.py deleted file mode 100644 index 4874c57a..00000000 --- a/tests/test_collecting/test_collector/test_collectors_merging/test_default_handler_adding.py +++ /dev/null @@ -1,15 +0,0 @@ -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_default_handler_after_including_into_another_collector( - default_handler, - handler_as_function, -): - collector1 = Collector() - collector2 = Collector(default=default_handler) - - collector1.include_collector(collector2) - - assert collector1.default_message_handler == collector2.default_message_handler diff --git a/tests/test_collecting/test_collector/test_collectors_merging/test_dependencies_order.py b/tests/test_collecting/test_collector/test_collectors_merging/test_dependencies_order.py deleted file mode 100644 index 25c23cc3..00000000 --- a/tests/test_collecting/test_collector/test_collectors_merging/test_dependencies_order.py +++ /dev/null @@ -1,93 +0,0 @@ -from typing import Callable - -import pytest - -from botx import Collector, Depends - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def build_botx_dependency(number: int) -> Callable[[], None]: - def factory(): - """Just do nothing.""" - - factory.number = number - return factory - - -@pytest.fixture() -def build_dependency(): - return build_botx_dependency - - -def test_preserving_order_after_merging(message, handler_as_function, build_dependency): - message.command.body = "/command" - - collector1 = Collector(dependencies=[Depends(build_dependency(1))]) - collector2 = Collector(dependencies=[Depends(build_dependency(2))]) - - collector2.add_handler( - handler=handler_as_function, - body=message.command.command, - dependencies=[Depends(build_dependency(3))], - ) - - collector1.include_collector(collector2) - - handler = collector1.handler_for("handler_function") - - numbers = [dep.dependency.number for dep in handler.dependencies] - - assert numbers == [1, 2, 3] - - -def test_preserving_order_after_merging_for_default_handler( - message, - default_handler, - build_dependency, -): - message.command.body = "/command" - - default_handler.dependencies = [Depends(build_dependency(3))] - - collector1 = Collector(dependencies=[Depends(build_dependency(1))]) - collector2 = Collector( - dependencies=[Depends(build_dependency(2))], - default=default_handler, - ) - - collector1.include_collector(collector2) - - handler = collector1.default_message_handler - - numbers = [dep.dependency.number for dep in handler.dependencies] - - assert numbers == [1, 2, 3] - - -def test_dependencies_order_in_include_collector( - message, - handler_as_function, - build_dependency, -): - message.command.body = "/command" - - collector1 = Collector() - collector2 = Collector() - - collector2.add_handler( - handler=handler_as_function, - body=message.command.command, - dependencies=[Depends(build_dependency(2))], - ) - - collector1.include_collector( - collector2, - dependencies=[Depends(build_dependency(1))], - ) - - handler = collector1.handler_for("handler_function") - - numbers = [dep.dependency.number for dep in handler.dependencies] - - assert numbers == [1, 2] diff --git a/tests/test_collecting/test_collector/test_collectors_merging/test_errors.py b/tests/test_collecting/test_collector/test_collectors_merging/test_errors.py deleted file mode 100644 index 35bc1655..00000000 --- a/tests/test_collecting/test_collector/test_collectors_merging/test_errors.py +++ /dev/null @@ -1,51 +0,0 @@ -import pytest - -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_error_when_merging_handlers_with_equal_bodies(build_handler): - collector1 = Collector() - collector1.add_handler( - handler=build_handler("handler1"), - body="/body", - name="handler1", - ) - - collector2 = Collector() - collector2.add_handler( - handler=build_handler("handler2"), - body="/body", - name="handler2", - ) - - with pytest.raises(AssertionError): - collector1.include_collector(collector2) - - -def test_error_when_merging_handlers_with_equal_names(build_handler): - collector1 = Collector() - collector1.add_handler( - handler=build_handler("handler1"), - body="/body1", - name="handler", - ) - - collector2 = Collector() - collector2.add_handler( - handler=build_handler("handler2"), - body="/body2", - name="handler", - ) - - with pytest.raises(AssertionError): - collector1.include_collector(collector2) - - -def test_only_single_default_handler_can_defined_in_collector(default_handler): - collector1 = Collector(default=default_handler) - collector2 = Collector(default=default_handler) - - with pytest.raises(AssertionError): - collector1.include_collector(collector2) diff --git a/tests/test_collecting/test_collector/test_commands_generation.py b/tests/test_collecting/test_collector/test_commands_generation.py deleted file mode 100644 index 4e515739..00000000 --- a/tests/test_collecting/test_collector/test_commands_generation.py +++ /dev/null @@ -1,29 +0,0 @@ -import pytest - -from botx.exceptions import NoMatchFound - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_error_when_no_args_passed(collector_cls): - collector = collector_cls() - with pytest.raises(TypeError): - collector.command_for() - - -def test_building_command_with_arguments(handler_as_function, collector_cls): - collector = collector_cls() - collector.add_handler(handler=handler_as_function, name="handler", body="/handler") - built_command = collector.command_for("handler", "arg1", 1, True) - assert built_command == "/handler arg1 1 True" - - -def test_raising_exception_when_generating_command_and_not_found( - build_handler_for_collector, - collector_cls, -): - collector = collector_cls() - collector.handler(build_handler_for_collector("handler1")) - collector.handler(build_handler_for_collector("handler2")) - with pytest.raises(NoMatchFound): - collector.command_for("not-existing-handler") diff --git a/tests/test_collecting/test_collector/test_decorators/__init__.py b/tests/test_collecting/test_collector/test_decorators/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_collecting/test_collector/test_decorators/test_default.py b/tests/test_collecting/test_collector/test_decorators/test_default.py deleted file mode 100644 index 966510c7..00000000 --- a/tests/test_collecting/test_collector/test_decorators/test_default.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_defining_default_handler_in_collector_as_decorator( - handler_as_function, - extract_collector, - collector_cls, -): - collector = collector_cls() - collector.default()(handler_as_function) - assert extract_collector(collector).default_message_handler - - -def test_error_when_default_already_exists( - handler_as_function, - extract_collector, - collector_cls, -): - collector = collector_cls() - collector.default()(handler_as_function) - - with pytest.raises(AssertionError): - collector.default()(handler_as_function) diff --git a/tests/test_collecting/test_collector/test_decorators/test_handler.py b/tests/test_collecting/test_collector/test_decorators/test_handler.py deleted file mode 100644 index eba5d870..00000000 --- a/tests/test_collecting/test_collector/test_decorators/test_handler.py +++ /dev/null @@ -1,14 +0,0 @@ -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_defining_handler_in_collector_as_decorator( - handler_as_function, - extract_collector, - collector_cls, -): - collector = Collector() - collector.handler()(handler_as_function) - handlers = [collector.handler_for("handler_function")] - assert handlers diff --git a/tests/test_collecting/test_collector/test_decorators/test_hidden.py b/tests/test_collecting/test_collector/test_decorators/test_hidden.py deleted file mode 100644 index 613debb1..00000000 --- a/tests/test_collecting/test_collector/test_decorators/test_hidden.py +++ /dev/null @@ -1,11 +0,0 @@ -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_defining_hidden_handler_in_collector_as_decorator( - handler_as_function, - extract_collector, - collector_cls, -): - collector = collector_cls() - collector.hidden()(handler_as_function) - assert not collector.handlers[0].include_in_status diff --git a/tests/test_collecting/test_collector/test_decorators/test_system_event.py b/tests/test_collecting/test_collector/test_decorators/test_system_event.py deleted file mode 100644 index c72edcba..00000000 --- a/tests/test_collecting/test_collector/test_decorators/test_system_event.py +++ /dev/null @@ -1,64 +0,0 @@ -import pytest - -from botx import SystemEvents - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_registration_handler_for_several_system_events( - handler_as_function, - extract_collector, - collector_cls, -): - system_events = { - SystemEvents.chat_created, - SystemEvents.file_transfer, - SystemEvents.added_to_chat, - SystemEvents.deleted_from_chat, - SystemEvents.left_from_chat, - SystemEvents.internal_bot_notification, - SystemEvents.cts_login, - SystemEvents.cts_logout, - SystemEvents.smartapp_event, - } - collector = collector_cls() - collector.system_event( - handler=handler_as_function, - events=list(system_events), - ) - handlers = [SystemEvents(handler.body) for handler in collector.handlers] - assert handlers - - -@pytest.mark.parametrize( - "event", - [ - SystemEvents.added_to_chat, - SystemEvents.deleted_from_chat, - SystemEvents.chat_created, - SystemEvents.file_transfer, - SystemEvents.left_from_chat, - SystemEvents.cts_login, - SystemEvents.cts_logout, - SystemEvents.smartapp_event, - ], -) -def test_defining_system_handler_in_collector_as_decorator( - handler_as_function, - extract_collector, - collector_cls, - event, -): - collector = collector_cls() - getattr(collector, event.name)()(handler_as_function) - assert SystemEvents(collector.handlers[0].body) == event - - -def test_error_when_no_event_was_passed( - handler_as_function, - extract_collector, - collector_cls, -): - collector = collector_cls() - with pytest.raises(AssertionError): - collector.system_event(handler=handler_as_function) diff --git a/tests/test_collecting/test_collector/test_execution.py b/tests/test_collecting/test_collector/test_execution.py deleted file mode 100644 index 1f04359b..00000000 --- a/tests/test_collecting/test_collector/test_execution.py +++ /dev/null @@ -1,45 +0,0 @@ -import threading - -import pytest - -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) -pytestmark = pytest.mark.asyncio - - -async def test_execution_when_full_match(message, build_handler): - event = threading.Event() - message.command.body = "/command" - - collector = Collector() - collector.add_handler(build_handler(event), body=message.command.body) - - await collector.handle_message(message) - - assert event.is_set() - - -async def test_executing_handler_when_partial_match(message, build_handler): - event = threading.Event() - message.command.body = "/command with arguments" - - collector = Collector() - collector.add_handler(build_handler(event), body=message.command.command) - - await collector.handle_message(message) - - assert event.is_set() - - -async def test_execution_internal_bot_notification( - internal_bot_notification_message, - build_handler, -): - event = threading.Event() - collector = Collector() - collector.internal_bot_notification(build_handler(event)) - - await collector.handle_message(internal_bot_notification_message) - - assert event.is_set() diff --git a/tests/test_collecting/test_collector/test_handler_definition_errors.py b/tests/test_collecting/test_collector/test_handler_definition_errors.py deleted file mode 100644 index b07dc764..00000000 --- a/tests/test_collecting/test_collector/test_handler_definition_errors.py +++ /dev/null @@ -1,12 +0,0 @@ -import pytest -from pydantic import ValidationError - -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_handler_can_not_consist_from_slashes_only(handler_as_function): - collector = Collector() - with pytest.raises(ValidationError): - collector.add_handler(handler_as_function, body="////") diff --git a/tests/test_collecting/test_collector/test_handlers_definition.py b/tests/test_collecting/test_collector/test_handlers_definition.py deleted file mode 100644 index 896313bb..00000000 --- a/tests/test_collecting/test_collector/test_handlers_definition.py +++ /dev/null @@ -1,9 +0,0 @@ -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_collector_default_handler_generating(default_handler): - collector = Collector(default=default_handler) - - assert collector.default_message_handler == default_handler diff --git a/tests/test_collecting/test_collector/test_handlers_order.py b/tests/test_collecting/test_collector/test_handlers_order.py deleted file mode 100644 index 92f2c6c3..00000000 --- a/tests/test_collecting/test_collector/test_handlers_order.py +++ /dev/null @@ -1,37 +0,0 @@ -import pytest - -from botx import Collector - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -@pytest.fixture() -def collector_with_handlers(handler_as_function): - collector = Collector() - for index in range(1, 31): - body = "/{0}".format("a" * index) - collector.add_handler(handler=handler_as_function, body=body, name=str(index)) - - return collector - - -def test_sorting_handlers_in_collector_by_body_length(collector_with_handlers): - added_handlers = collector_with_handlers.sorted_handlers - assert added_handlers == sorted( - added_handlers, - key=lambda handler: len(handler.body), - reverse=True, - ) - - -def test_preserve_length_sort_when_merging_collectors( - collector_with_handlers, - handler_as_function, -): - collector = Collector() - collector.add_handler(handler=handler_as_function, body="/{0}".format("a" * 1000)) - - collector_with_handlers.include_collector(collector) - - added_handlers = collector_with_handlers.sorted_handlers - assert added_handlers[0] == collector.handlers[0] diff --git a/tests/test_collecting/test_collector/test_handlers_search.py b/tests/test_collecting/test_collector/test_handlers_search.py deleted file mode 100644 index cad298d6..00000000 --- a/tests/test_collecting/test_collector/test_handlers_search.py +++ /dev/null @@ -1,11 +0,0 @@ -import pytest - -from botx.exceptions import NoMatchFound - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_raising_exception_when_searching_for_handler_and_no_found(collector_cls): - collector = collector_cls() - with pytest.raises(NoMatchFound): - collector.handler_for("not-existing-handler") diff --git a/tests/test_collecting/test_handler/__init__.py b/tests/test_collecting/test_handler/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_collecting/test_handler/test_attributes.py b/tests/test_collecting/test_handler/test_attributes.py deleted file mode 100644 index a86267e2..00000000 --- a/tests/test_collecting/test_handler/test_attributes.py +++ /dev/null @@ -1,22 +0,0 @@ -from botx.collecting.handlers.handler import Handler - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_handler_docstring_stored_as_full_description(handler_as_function): - handler = Handler(body="/command", handler=handler_as_function) - assert handler.full_description == handler_as_function.__doc__ - - -def test_handler_from_function(handler_as_function): - handler = Handler(body="/command", handler=handler_as_function) - assert handler.name == "handler_function" - - -def test_name_when_name_was_passed_explicitly(handler_as_function): - handler = Handler(handler=handler_as_function, body="/command", name="my_handler") - assert handler.name == "my_handler" - - -def test_any_body_for_hidden_handler(handler_as_function): - Handler(handler=handler_as_function, body="any text!", include_in_status=False) diff --git a/tests/test_collecting/test_handler/test_commands_generation.py b/tests/test_collecting/test_handler/test_commands_generation.py deleted file mode 100644 index 1ee5ed8b..00000000 --- a/tests/test_collecting/test_handler/test_commands_generation.py +++ /dev/null @@ -1,13 +0,0 @@ -from botx.collecting.handlers.handler import Handler - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_no_extra_space_on_command_built_through_command_for(handler_as_function): - handler = Handler(body="/command", handler=handler_as_function) - - assert handler.command_for() == "/command" - - built_command_with_args = handler.command_for(None, 1, "some string", True) - - assert built_command_with_args == "/command 1 some string True" diff --git a/tests/test_collecting/test_handler/test_constructing_errors.py b/tests/test_collecting/test_handler/test_constructing_errors.py deleted file mode 100644 index 3dbfd50b..00000000 --- a/tests/test_collecting/test_handler/test_constructing_errors.py +++ /dev/null @@ -1,31 +0,0 @@ -import pytest -from pydantic import ValidationError - -from botx.collecting.handlers.handler import Handler - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_error_handler_from_class(handler_as_class): - with pytest.raises(ValidationError): - Handler(body="/command", handler=handler_as_class) - - -def test_error_from_callable(handler_as_callable_object): - with pytest.raises(ValidationError): - Handler(body="/command", handler=handler_as_callable_object) - - -def test_slash_in_command_for_public_command(handler_as_function): - with pytest.raises(ValidationError): - Handler(body="command", handler=handler_as_function) - - -def test_only_one_slash_in_public_command(handler_as_function): - with pytest.raises(ValidationError): - Handler(body="//command", handler=handler_as_function) - - -def test_that_menu_command_contain_only_single_word(handler_as_function): - with pytest.raises(ValidationError): - Handler(body="/many words handler", handler=handler_as_function) diff --git a/tests/test_collecting/test_handler/test_equeality.py b/tests/test_collecting/test_handler/test_equeality.py deleted file mode 100644 index e6893a1d..00000000 --- a/tests/test_collecting/test_handler/test_equeality.py +++ /dev/null @@ -1,20 +0,0 @@ -from botx.collecting.handlers.handler import Handler - -pytest_plugins = ("tests.test_collecting.fixtures",) - - -def test_equality_is_false_if_not_handler_passed(handler_as_function): - handler = Handler(body="/command", handler=handler_as_function) - assert handler != "" - - -def test_equality_is_false_if_handlers_are_different(handler_as_function): - handler1 = Handler(body="/command1", handler=handler_as_function) - handler2 = Handler(body="/command2", handler=handler_as_function) - assert handler1 != handler2 - - -def test_equality_if_handlers_are_similar(handler_as_function): - handler1 = Handler(body="/command", handler=handler_as_function) - handler2 = Handler(body="/command", handler=handler_as_function) - assert handler1 == handler2 diff --git a/tests/test_dependencies/__init__.py b/tests/test_dependencies/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_dependencies/test_default_handler_dependencies.py b/tests/test_dependencies/test_default_handler_dependencies.py deleted file mode 100644 index 3c5aa631..00000000 --- a/tests/test_dependencies/test_default_handler_dependencies.py +++ /dev/null @@ -1,51 +0,0 @@ -import threading - -import pytest - -from botx import Collector, Depends - -pytestmark = pytest.mark.asyncio - - -async def test_dependencies_from_bot_on_default_handler( - bot, - incoming_message, - client, - build_handler, -): - event1 = threading.Event() - event2 = threading.Event() - - bot.collector = bot.exception_middleware.executor = Collector( - dependencies=[Depends(build_handler(event1))], - ) - - bot.default(build_handler(event2)) - - await client.send_command(incoming_message) - - assert event1.is_set() - assert event2.is_set() - - -async def test_dependency_saved_after_include_collector( - bot, - incoming_message, - client, - build_handler, -): - event1 = threading.Event() - event2 = threading.Event() - - collector = Collector() - collector.default(build_handler(event1)) - - bot.collector = bot.exception_middleware.executor = Collector( - dependencies=[Depends(build_handler(event2))], - ) - bot.include_collector(collector) - - await client.send_command(incoming_message) - - assert event1.is_set() - assert event2.is_set() diff --git a/tests/test_dependencies/test_dependencies_cache.py b/tests/test_dependencies/test_dependencies_cache.py deleted file mode 100644 index 60895ac8..00000000 --- a/tests/test_dependencies/test_dependencies_cache.py +++ /dev/null @@ -1,42 +0,0 @@ -import threading -from typing import Callable - -import pytest - -from botx import Depends - -pytestmark = pytest.mark.asyncio - - -def build_botx_dependency(lock: threading.Lock) -> Callable[[], None]: - def factory(): - lock.acquire() - - return factory - - -@pytest.fixture() -def build_dependency(): - return build_botx_dependency - - -async def test_dependency_executed_only_once_per_message( - bot, - client, - incoming_message, - build_dependency, - build_handler, -): - event = threading.Event() - lock = threading.Lock() - - dependency_function = build_dependency(lock) - - bot.default( - build_handler(event), - dependencies=[Depends(dependency_function), Depends(dependency_function)], - ) - - await client.send_command(incoming_message) - - assert event.is_set() diff --git a/tests/test_dependencies/test_errors.py b/tests/test_dependencies/test_errors.py deleted file mode 100644 index 13a80dce..00000000 --- a/tests/test_dependencies/test_errors.py +++ /dev/null @@ -1,12 +0,0 @@ -import pytest - -from botx.collecting.handlers.handler import Handler -from botx.dependencies.solving import get_executor - - -def test_error_when_creating_executor_without_call(build_handler): - handler = Handler(build_handler(...), body="/body") - dependant = handler.dependant - dependant.call = None - with pytest.raises(AssertionError): - get_executor(dependant) diff --git a/tests/test_dependencies/test_flow_control.py b/tests/test_dependencies/test_flow_control.py deleted file mode 100644 index 3ddcf012..00000000 --- a/tests/test_dependencies/test_flow_control.py +++ /dev/null @@ -1,51 +0,0 @@ -import threading -from typing import Callable - -import pytest - -from botx import DependencyFailure, Depends - -pytestmark = pytest.mark.asyncio - - -def build_botx_fail_dependency(event: threading.Event) -> Callable[[], None]: - def factory(): - event.set() - raise DependencyFailure - - return factory - - -@pytest.fixture() -def build_fail_dependency(): - return build_botx_fail_dependency - - -async def test_flow_stop_if_error_raised( - bot, - client, - incoming_message, - build_handler, - build_fail_dependency, -): - handler_event = threading.Event() - - dependency_event1 = threading.Event() - dependency_event2 = threading.Event() - dependency_event3 = threading.Event() - - bot.default( - handler=build_handler(handler_event), - dependencies=[ - Depends(build_handler(dependency_event1)), - Depends(build_botx_fail_dependency(dependency_event2)), - Depends(build_handler(dependency_event3)), - ], - ) - - await client.send_command(incoming_message) - - assert dependency_event1.is_set() - assert dependency_event2.is_set() - assert not dependency_event3.is_set() - assert not handler_event.is_set() diff --git a/tests/test_dependencies/test_overrides.py b/tests/test_dependencies/test_overrides.py deleted file mode 100644 index ecb8fa38..00000000 --- a/tests/test_dependencies/test_overrides.py +++ /dev/null @@ -1,81 +0,0 @@ -import threading - -import pytest - -from botx import Depends - -pytestmark = pytest.mark.asyncio - - -async def test_that_dependency_can_be_overriden( - bot, - client, - incoming_message, - build_handler, -): - handler_event = threading.Event() - original_dependency_event = threading.Event() - fake_dependency_event = threading.Event() - - dependency = build_handler(original_dependency_event) - bot.default(build_handler(handler_event), dependencies=[Depends(dependency)]) - - bot.dependency_overrides[dependency] = build_handler(fake_dependency_event) - - await client.send_command(incoming_message) - - assert handler_event.is_set() - assert not original_dependency_event.is_set() - assert fake_dependency_event.is_set() - - -async def test_bot_is_used_as_default_provider( - bot, - client, - incoming_message, - build_handler, -): - handler_event = threading.Event() - original_dependency_event = threading.Event() - fake_dependency_event = threading.Event() - - dependency = build_handler(original_dependency_event) - bot.default( - build_handler(handler_event), - dependencies=[Depends(dependency)], - dependency_overrides_provider=None, - ) - - bot.dependency_overrides[dependency] = build_handler(fake_dependency_event) - - await client.send_command(incoming_message) - - assert handler_event.is_set() - assert not original_dependency_event.is_set() - assert fake_dependency_event.is_set() - - -async def test_overrider_is_used_if_not_none( - bot, - client, - incoming_message, - build_handler, -): - handler_event = threading.Event() - original_dependency_event = threading.Event() - fake_dependency_event = threading.Event() - - dependency = build_handler(original_dependency_event) - bot.default( - build_handler(handler_event), - dependencies=[Depends(dependency)], - dependency_overrides_provider={}, - ) - - bot.dependency_overrides[dependency] = build_handler(fake_dependency_event) - - await client.send_command(incoming_message) - - assert handler_event.is_set() - assert not fake_dependency_event.is_set() - assert original_dependency_event.is_set() diff --git a/tests/test_dependencies/test_special_params/__init__.py b/tests/test_dependencies/test_special_params/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_dependencies/test_special_params/test_definition/__init__.py b/tests/test_dependencies/test_special_params/test_definition/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_dependencies/test_special_params/test_definition/test_async_client_param.py b/tests/test_dependencies/test_special_params/test_definition/test_async_client_param.py deleted file mode 100644 index cc912690..00000000 --- a/tests/test_dependencies/test_special_params/test_definition/test_async_client_param.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from botx import AsyncClient - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(client: AsyncClient) -> None: - storage.client = client - - return factory - - -async def test_passing_async_client_as_dependency( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - await client.send_command(incoming_message) - assert storage.client == bot.client diff --git a/tests/test_dependencies/test_special_params/test_definition/test_bot_param.py b/tests/test_dependencies/test_special_params/test_definition/test_bot_param.py deleted file mode 100644 index eb18cd1b..00000000 --- a/tests/test_dependencies/test_special_params/test_definition/test_bot_param.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from botx import Bot - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(bot: Bot) -> None: - storage.bot = bot - - return factory - - -async def test_passing_async_client_as_dependency( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - await client.send_command(incoming_message) - assert storage.bot == bot diff --git a/tests/test_dependencies/test_special_params/test_definition/test_depends_param.py b/tests/test_dependencies/test_special_params/test_definition/test_depends_param.py deleted file mode 100644 index 1db4167c..00000000 --- a/tests/test_dependencies/test_special_params/test_definition/test_depends_param.py +++ /dev/null @@ -1,29 +0,0 @@ -import pytest - -from botx import Depends - -pytestmark = pytest.mark.asyncio - - -def dependency(): - return 42 - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(dep: int = Depends(dependency)) -> None: # noqa: B008 - storage.dep = dep - - return factory - - -async def test_passing_async_client_as_dependency( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - await client.send_command(incoming_message) - assert storage.dep == 42 diff --git a/tests/test_dependencies/test_special_params/test_definition/test_message_param.py b/tests/test_dependencies/test_special_params/test_definition/test_message_param.py deleted file mode 100644 index 83aaac69..00000000 --- a/tests/test_dependencies/test_special_params/test_definition/test_message_param.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from botx import Message - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(message: Message) -> None: - storage.message = message - - return factory - - -async def test_passing_async_client_as_dependency( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - await client.send_command(incoming_message) - assert storage.message.incoming_message == incoming_message diff --git a/tests/test_dependencies/test_special_params/test_definition/test_sync_client_param.py b/tests/test_dependencies/test_special_params/test_definition/test_sync_client_param.py deleted file mode 100644 index 7bb3cd15..00000000 --- a/tests/test_dependencies/test_special_params/test_definition/test_sync_client_param.py +++ /dev/null @@ -1,25 +0,0 @@ -import pytest - -from botx import Client - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(client: Client) -> None: - storage.client = client - - return factory - - -async def test_passing_async_client_as_dependency( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - await client.send_command(incoming_message) - assert storage.client == bot.sync_client diff --git a/tests/test_dependencies/test_special_params/test_errors.py b/tests/test_dependencies/test_special_params/test_errors.py deleted file mode 100644 index 0c618520..00000000 --- a/tests/test_dependencies/test_special_params/test_errors.py +++ /dev/null @@ -1,14 +0,0 @@ -import pytest - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(_: int) -> None: - """Just do nothing""" - - return factory - - -def test_error_for_wrong_param(bot, handler_with_dependency) -> None: - with pytest.raises(ValueError): - bot.default(handler_with_dependency) diff --git a/tests/test_dependencies/test_special_params/test_forward_references_solving.py b/tests/test_dependencies/test_special_params/test_forward_references_solving.py deleted file mode 100644 index 3fbfd5bc..00000000 --- a/tests/test_dependencies/test_special_params/test_forward_references_solving.py +++ /dev/null @@ -1,27 +0,0 @@ -import pytest - -from botx import Message - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def handler_with_dependency(storage): - def factory(message: "Message") -> None: - storage.message = message - - return factory - - -async def test_solving_forward_references_for_special_dependencies( - bot, - client, - incoming_message, - handler_with_dependency, - storage, -): - bot.default(handler_with_dependency) - - await client.send_command(incoming_message) - - assert storage.message.incoming_message == incoming_message diff --git a/tests/test_docs/__init__.py b/tests/test_docs/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_end_to_end.py b/tests/test_end_to_end.py new file mode 100644 index 00000000..27abf5c5 --- /dev/null +++ b/tests/test_end_to_end.py @@ -0,0 +1,345 @@ +import os +from http import HTTPStatus +from typing import List +from uuid import UUID + +import httpx +import pytest +from dotenv import load_dotenv +from fastapi import APIRouter, Depends, FastAPI, Request +from fastapi.responses import JSONResponse +from fastapi.testclient import TestClient +from loguru import logger +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + UnknownBotAccountError, + build_bot_disabled_response, + build_command_accepted_response, +) + +# - Test utils - +load_dotenv() + + +def get_bot_accounts() -> List[BotAccountWithSecret]: + raw_credentials_list = os.getenv("BOT_CREDENTIALS") + if not raw_credentials_list: + raise RuntimeError("BOT_CREDENTIALS env not set") + + bot_accounts = [] + for raw_credentials in raw_credentials_list.split(","): + host, secret_key, raw_bot_id = raw_credentials.replace("|", "@").split("@") + bot_accounts.append( + BotAccountWithSecret( + id=UUID(raw_bot_id), + host=host, + secret_key=secret_key, + ), + ) + + return bot_accounts + + +# - Bot setup - +collector = HandlerCollector() + + +@collector.command("/debug", description="Simple debug command") +async def debug_handler(message: IncomingMessage, bot: Bot) -> None: + await bot.answer_message("Hi!") + + +def bot_factory( + bot_accounts: List[BotAccountWithSecret], +) -> Bot: + return Bot(collectors=[collector], bot_accounts=bot_accounts) + + +# - FastAPI integration - +def get_bot(request: Request) -> Bot: + assert isinstance(request.app.state.bot, Bot) + + return request.app.state.bot + + +bot_dependency = Depends(get_bot) + +router = APIRouter() + + +@router.post("/command") +async def command_handler( + request: Request, + bot: Bot = bot_dependency, +) -> JSONResponse: + try: + bot.async_execute_raw_bot_command(await request.json()) + except ValueError: + error_label = "Bot command validation error" + logger.exception(error_label) + + return JSONResponse( + build_bot_disabled_response(error_label), + status_code=HTTPStatus.SERVICE_UNAVAILABLE, + ) + except UnknownBotAccountError as exc: + error_label = f"No credentials for bot {exc.bot_id}" + logger.warning(error_label) + + return JSONResponse( + build_bot_disabled_response(error_label), + status_code=HTTPStatus.SERVICE_UNAVAILABLE, + ) + + return JSONResponse( + build_command_accepted_response(), + status_code=HTTPStatus.ACCEPTED, + ) + + +@router.get("/status") +async def status_handler(request: Request, bot: Bot = bot_dependency) -> JSONResponse: + status = await bot.raw_get_status(dict(request.query_params)) + return JSONResponse(status) + + +@router.post("/notification/callback") +async def callback_handler( + request: Request, + bot: Bot = bot_dependency, +) -> JSONResponse: + bot.set_raw_botx_method_result(await request.json()) + return JSONResponse( + build_command_accepted_response(), + status_code=HTTPStatus.ACCEPTED, + ) + + +def fastapi_factory(bot: Bot) -> FastAPI: + application = FastAPI() + application.state.bot = bot + + application.add_event_handler("startup", bot.startup) + application.add_event_handler("shutdown", bot.shutdown) + + application.include_router(router) + + return application + + +# https://www.uvicorn.org/#application-factories +def asgi_factory() -> FastAPI: + bot_accounts = get_bot_accounts() + bot = bot_factory(bot_accounts) + return fastapi_factory(bot) + + +# - Tests - +@pytest.fixture +def bot(bot_account: BotAccountWithSecret) -> Bot: + return bot_factory(bot_accounts=[bot_account]) + + +pytestmark = [ + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +def test__web_app__bot_status( + bot_id: UUID, + bot: Bot, +) -> None: + # - Arrange - + query_params = { + "bot_id": str(bot_id), + "chat_type": "chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + # - Act - + with TestClient(fastapi_factory(bot)) as test_client: + response = test_client.get( + "/status", + params=query_params, + ) + + # - Assert - + assert response.status_code == HTTPStatus.OK + assert response.json() == { + "result": { + "commands": [ + { + "body": "/debug", + "description": "Simple debug command", + "name": "/debug", + }, + ], + "enabled": True, + "status_message": "Bot is working", + }, + "status": "ok", + } + + +def test__web_app__bot_command( + respx_mock: MockRouter, + bot_id: UUID, + host: str, + bot: Bot, +) -> None: + # - Arrange - + direct_notification_endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + command_payload = { + "bot_id": str(bot_id), + "command": { + "body": "/debug", + "command_type": "user", + "data": {}, + "metadata": {}, + }, + "attachments": [], + "async_files": [], + "entities": [], + "source_sync_id": None, + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": False, + "timezone": "Europe/Moscow", + }, + "device_software": None, + "group_chat_id": "30dc1980-643a-00ad-37fc-7cc10d74e935", + "host": "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + "username": None, + }, + "proto_version": 4, + } + + callback_payload = { + "status": "ok", + "sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3", + "result": {}, + } + + # - Act - + with TestClient(fastapi_factory(bot)) as test_client: + command_response = test_client.post( + "/command", + json=command_payload, + ) + + callback_response = test_client.post( + "/notification/callback", + json=callback_payload, + ) + + # - Assert - + assert command_response.status_code == HTTPStatus.ACCEPTED + assert direct_notification_endpoint.called + assert callback_response.status_code == HTTPStatus.ACCEPTED + + +def test__web_app__unknown_bot_response( + bot: Bot, +) -> None: + # - Arrange - + payload = { + "bot_id": "c755e147-30a5-45df-b46a-c75aa6089c8f", + "command": { + "body": "/debug", + "command_type": "user", + "data": {}, + "metadata": {}, + }, + "attachments": [], + "async_files": [], + "entities": [], + "source_sync_id": None, + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": False, + "timezone": "Europe/Moscow", + }, + "device_software": None, + "group_chat_id": "30dc1980-643a-00ad-37fc-7cc10d74e935", + "host": "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + "username": None, + }, + "proto_version": 4, + } + + # - Act - + with TestClient(fastapi_factory(bot)) as test_client: + response = test_client.post( + "/command", + json=payload, + ) + + # - Assert - + assert response.status_code == HTTPStatus.SERVICE_UNAVAILABLE + + +def test__web_app__disabled_bot_response( + bot: Bot, +) -> None: + # - Arrange - + payload = {"incorrect": "request"} + + # - Act - + with TestClient(fastapi_factory(bot)) as test_client: + response = test_client.post( + "/command", + json=payload, + ) + + # - Assert - + assert response.status_code == HTTPStatus.SERVICE_UNAVAILABLE + assert response.json() == { + "error_data": {"status_message": "Bot command validation error"}, + "errors": [], + "reason": "bot_disabled", + } diff --git a/tests/test_exception_handlers/__init__.py b/tests/test_exception_handlers/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_exception_handlers/test_logging.py b/tests/test_exception_handlers/test_logging.py deleted file mode 100644 index 7e2b9e88..00000000 --- a/tests/test_exception_handlers/test_logging.py +++ /dev/null @@ -1,18 +0,0 @@ -import pytest - -pytestmark = pytest.mark.asyncio - - -async def test_logging_that_handler_was_not_found( - bot, - client, - incoming_message, - loguru_caplog, -) -> None: - await client.send_command(incoming_message) - - error_message = "handler for {0!r} was not found".format( - incoming_message.command.body, - ) - - assert error_message in loguru_caplog.text diff --git a/tests/test_exception_middleware.py b/tests/test_exception_middleware.py new file mode 100644 index 00000000..ede3fe51 --- /dev/null +++ b/tests/test_exception_middleware.py @@ -0,0 +1,112 @@ +import asyncio +from typing import Callable +from unittest.mock import MagicMock, call + +import pytest + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__exception_middleware__handler_called( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + exc = ValueError("test_error") + value_error_handler = MagicMock(asyncio.Future()) + + user_command = incoming_message_factory(body="/command") + + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + raise exc + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + exception_handlers={ValueError: value_error_handler}, + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert len(value_error_handler.mock_calls) == 1 + assert value_error_handler.mock_calls[0] == call(user_command, built_bot, exc) + + +async def test__exception_middleware__without_handler_logs( + incoming_message_factory: Callable[..., IncomingMessage], + loguru_caplog: pytest.LogCaptureFixture, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + raise ValueError("Testing exception middleware") + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert "Uncaught exception ValueError" in loguru_caplog.text + assert "Testing exception middleware" in loguru_caplog.text + + +async def test__exception_middleware__error_in_handler_logs( + incoming_message_factory: Callable[..., IncomingMessage], + loguru_caplog: pytest.LogCaptureFixture, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + async def exception_handler( + message: IncomingMessage, + bot: Bot, + exc: Exception, + ) -> None: + raise ValueError("Nested error") + + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + raise ValueError("Testing exception middleware") + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + exception_handlers={Exception: exception_handler}, + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert "Uncaught exception ValueError in exception handler" in loguru_caplog.text + assert "Testing exception middleware" in loguru_caplog.text + assert "Nested error" in loguru_caplog.text diff --git a/tests/test_exceptions/__init__.py b/tests/test_exceptions/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_exceptions/test_botx_exception.py b/tests/test_exceptions/test_botx_exception.py deleted file mode 100644 index e51034a0..00000000 --- a/tests/test_exceptions/test_botx_exception.py +++ /dev/null @@ -1,8 +0,0 @@ -from botx.exceptions import BotXException - - -def test_to_string_fills_template(): - exc = BotXException(arg=42) - exc.message_template = "template with {arg}" - - assert str(exc) == "template with 42" diff --git a/tests/test_exceptions/test_route_deprecated_error.py b/tests/test_exceptions/test_route_deprecated_error.py deleted file mode 100644 index 2b2bb6c9..00000000 --- a/tests/test_exceptions/test_route_deprecated_error.py +++ /dev/null @@ -1,27 +0,0 @@ -import uuid - -import httpx -import pytest - -from botx.clients.methods.v3.chats.info import Info -from botx.concurrency import callable_to_coroutine -from botx.exceptions import BotXAPIRouteDeprecated - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_clients.fixtures",) - - -async def test_raising_route_deprecated_error(client, requests_client): - method = Info(host="example.com", group_chat_id=uuid.uuid4()) - errors_to_raise = {Info: (httpx.codes.GONE, {})} - - with client.error_client(errors=errors_to_raise): - request = requests_client.build_request(method) - response = await callable_to_coroutine(requests_client.execute, request) - - with pytest.raises(BotXAPIRouteDeprecated): - await callable_to_coroutine( - requests_client.process_response, - method, - response, - ) diff --git a/tests/test_exceptions/test_routing_match_error.py b/tests/test_exceptions/test_routing_match_error.py deleted file mode 100644 index 5184653e..00000000 --- a/tests/test_exceptions/test_routing_match_error.py +++ /dev/null @@ -1,13 +0,0 @@ -import pytest - -from botx.exceptions import NoMatchFound - -pytestmark = pytest.mark.asyncio - - -async def test_search_param_in_matching_error(bot, message, client): - with pytest.raises(NoMatchFound) as err_info: - await bot.collector.handle_message(message) - - error = err_info.value - assert error.search_param == message.command.body diff --git a/tests/test_exceptions/test_unknown_server.py b/tests/test_exceptions/test_unknown_server.py deleted file mode 100644 index cff8d424..00000000 --- a/tests/test_exceptions/test_unknown_server.py +++ /dev/null @@ -1,15 +0,0 @@ -import pytest - -from botx import UnknownBotError - -pytestmark = pytest.mark.asyncio - - -async def test_host_in_server_error(bot, incoming_message, client): - bot.bot_accounts = [] - - with pytest.raises(UnknownBotError) as err_info: - await client.send_command(incoming_message) - - error = err_info.value - assert error.bot_id == incoming_message.bot_id diff --git a/tests/test_files.py b/tests/test_files.py new file mode 100644 index 00000000..1274e43d --- /dev/null +++ b/tests/test_files.py @@ -0,0 +1,237 @@ +from http import HTTPStatus +from typing import Any, Callable, Dict, Optional +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import ( + AttachmentTypes, + Bot, + BotAccountWithSecret, + Document, + File, + HandlerCollector, + Image, + IncomingMessage, + Video, + Voice, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__async_file__open( + respx_mock: MockRouter, + host: str, + bot_account: BotAccountWithSecret, + bot_id: UUID, + api_incoming_message_factory: Callable[..., Dict[str, Any]], +) -> None: + # - Arrange - + endpoint = respx_mock.get( + f"https://{host}/api/v3/botx/files/download", + params={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + headers={"Authorization": "Bearer token"}, + ).mock( + return_value=httpx.Response( + HTTPStatus.OK, + content=b"Hello, world!\n", + ), + ) + + payload = api_incoming_message_factory( + bot_id=bot_id, + async_file={ + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "c3b9def2-b2c8-4732-b61f-99b9b110fa80", + }, + group_chat_id="054af49e-5e18-4dca-ad73-4f96b6de63fa", + host=host, + ) + + collector = HandlerCollector() + read_content: Optional[bytes] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal read_content + + assert message.file + async with message.file.open() as fo: + read_content = await fo.read() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert read_content == b"Hello, world!\n" + assert endpoint.called + + +API_AND_DOMAIN_FILES = ( + ( + { + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + Image( + type=AttachmentTypes.IMAGE, + filename="pass.png", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="image/png", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ), + ( + { + "type": "video", + "file": "https://link.to/file", + "file_mime_type": "video/mp4", + "file_name": "pass.mp4", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "duration": 10, + }, + Video( + type=AttachmentTypes.VIDEO, + filename="pass.mp4", + size=1502345, + is_async_file=True, + duration=10, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="video/mp4", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ), + ( + { + "type": "document", + "file": "https://link.to/file", + "file_mime_type": "plain/text", + "file_name": "pass.txt", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + Document( + type=AttachmentTypes.DOCUMENT, + filename="pass.txt", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="plain/text", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ), + ( + { + "type": "voice", + "file": "https://link.to/file", + "file_mime_type": "audio/mp3", + "file_name": "pass.mp3", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + "duration": 10, + }, + Voice( + type=AttachmentTypes.VOICE, + filename="pass.mp3", + size=1502345, + is_async_file=True, + duration=10, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="audio/mp3", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + ), +) + + +@pytest.mark.parametrize( + "api_async_file,domain_async_file", + API_AND_DOMAIN_FILES, +) +async def test__async_execute_raw_bot_command__different_file_types( + api_async_file: Dict[str, Any], + domain_async_file: File, + api_incoming_message_factory: Callable[..., Dict[str, Any]], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = api_incoming_message_factory(async_file=api_async_file) + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert incoming_message + assert incoming_message.file == domain_async_file diff --git a/tests/test_handler_collector.py b/tests/test_handler_collector.py new file mode 100644 index 00000000..3177318e --- /dev/null +++ b/tests/test_handler_collector.py @@ -0,0 +1,509 @@ +from typing import Callable +from unittest.mock import Mock + +import pytest + +from botx import ( + Bot, + BotAccountWithSecret, + ChatCreatedEvent, + HandlerCollector, + IncomingMessage, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +def test__handler_collector__command_with_space_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + with pytest.raises(ValueError) as exc: + + @collector.command("/ command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Assert - + assert "include space" in str(exc.value) + + +def test__handler_collector__command_without_leading_slash_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + with pytest.raises(ValueError) as exc: + + @collector.command("command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Assert - + assert "should start with '/'" in str(exc.value) + + +def test__handler_collector__visible_command_without_description_error_raised() -> None: + # - Act - + collector = HandlerCollector() + + with pytest.raises(ValueError) as exc: + + @collector.command("/command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Assert - + assert "Description is required" in str(exc.value) + + +def test__handler_collector__two_same_commands_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + + @collector.command("/command", description="My command") + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Assert - + assert "already registered" in str(exc.value) + assert "/command" in str(exc.value) + + +def test__handler_collector__two_default_handlers_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.default_message_handler + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + + @collector.default_message_handler + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Assert - + assert "already registered" in str(exc.value) + assert "Default" in str(exc.value) + + +def test__handler_collector__two_same_system_events_handlers_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.chat_created + async def handler_1(message: ChatCreatedEvent, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + + @collector.chat_created + async def handler_2(message: ChatCreatedEvent, bot: Bot) -> None: + pass + + # - Assert - + assert "already registered" in str(exc.value) + assert "Event" in str(exc.value) + + +def test___handler_collector__merge_collectors_with_same_command_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + other_collector = HandlerCollector() + + @other_collector.command("/command", description="My command") + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + collector.include(other_collector) + + # - Assert - + assert "already registered" in str(exc.value) + assert "/command" in str(exc.value) + + +def test__handler_collector__merge_collectors_with_default_handlers_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.default_message_handler + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + other_collector = HandlerCollector() + + @other_collector.default_message_handler + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + collector.include(other_collector) + + # - Assert - + assert "already registered" in str(exc.value) + assert "Default" in str(exc.value) + + +def test__handler_collector__merge_collectors_with_same_system_events_handlers_error_raised() -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.chat_created + async def handler_1(message: ChatCreatedEvent, bot: Bot) -> None: + pass + + other_collector = HandlerCollector() + + @other_collector.chat_created + async def handler_2(message: ChatCreatedEvent, bot: Bot) -> None: + pass + + # - Act - + with pytest.raises(ValueError) as exc: + collector.include(other_collector) + + # - Assert - + assert "already registered" in str(exc.value) + assert "event" in str(exc.value) + + +@pytest.mark.asyncio +async def test__handler_collector__command_handler_called( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__unicode_command_error_raised( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + russian_command = incoming_message_factory(body="/команда") + collector = HandlerCollector() + + @collector.command("/команда", description="Моя команда") + async def handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(russian_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__correct_command_handler_called( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + incorrect_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def correct_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + @collector.command("/other", description="My command") + async def incorrect_handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + incorrect_handler_trigger.assert_not_called() + + +@pytest.mark.asyncio +async def test__handler_collector__correct_command_handler_called_in_merged_collectors( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + incorrect_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + collector_1 = HandlerCollector() + collector_2 = HandlerCollector() + + @collector_1.command("/command", description="My command") + async def correct_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + @collector_2.command("/command-two", description="My command") + async def incorrect_handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector_1, collector_2], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + incorrect_handler_trigger.assert_not_called() + + +@pytest.mark.asyncio +async def test__handler_collector__default_handler_called( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + collector = HandlerCollector() + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__empty_command_goes_to_default_handler( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + empty_command = incoming_message_factory(body="") + collector = HandlerCollector() + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(empty_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__invalid_command_goes_to_default_handler( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + empty_command = incoming_message_factory(body="/") + collector = HandlerCollector() + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(empty_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__handler_not_found_logged( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + collector = HandlerCollector() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert "`/command` not found" in loguru_caplog.text + + +@pytest.mark.asyncio +async def test__handler_collector__default_handler_in_first_collector_called( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + collector_1 = HandlerCollector() + collector_2 = HandlerCollector() + + @collector_1.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector_1, collector_2], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__default_handler_in_second_collector_called( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + collector_1 = HandlerCollector() + collector_2 = HandlerCollector() + + @collector_2.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector_1, collector_2], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +@pytest.mark.asyncio +async def test__handler_collector__handler_not_found_exception_logged( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + bot.async_execute_bot_command(incoming_message_factory(body="/command")) + await bot.shutdown() + + # - Assert - + assert "`/command` not found" in loguru_caplog.text + + +@pytest.mark.asyncio +async def test__handler_collector__handle_incoming_message_by_command_handler_not_found_exception_logged( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + collector = HandlerCollector() + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await collector.handle_incoming_message_by_command( + incoming_message_factory(body="Text"), + bot, + command="/command", + ) + + # - Assert - + assert "`/command` not found" in loguru_caplog.text + + +@pytest.mark.asyncio +async def test__handler_collector__handle_incoming_message_by_command_succeed( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, + correct_handler_trigger: Mock, +) -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await collector.handle_incoming_message_by_command( + incoming_message_factory(body="Text"), + bot, + command="/command", + ) + + # - Assert - + correct_handler_trigger.assert_called_once() diff --git a/tests/test_incoming_message.py b/tests/test_incoming_message.py new file mode 100644 index 00000000..0284146a --- /dev/null +++ b/tests/test_incoming_message.py @@ -0,0 +1,475 @@ +from datetime import datetime +from typing import Callable, Optional +from uuid import UUID + +import pytest + +from botx import ( + AttachmentTypes, + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatTypes, + ClientPlatforms, + Forward, + HandlerCollector, + Image, + IncomingMessage, + Mention, + MentionList, + MentionTypes, + Reply, + UserDevice, + UserSender, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__async_execute_raw_bot_command__minimally_filled_incoming_message( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "/hello", + "command_type": "user", + "data": {}, + "metadata": {}, + }, + "attachments": [], + "async_files": [], + "entities": [], + "source_sync_id": None, + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "chat", + "device": None, + "device_meta": None, + "device_software": None, + "group_chat_id": "30dc1980-643a-00ad-37fc-7cc10d74e935", + "host": "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert incoming_message == IncomingMessage( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + sync_id=UUID("6f40a492-4b5f-54f3-87ee-77126d825b51"), + source_sync_id=None, + body="/hello", + data={}, + metadata={}, + sender=UserSender( + huid=UUID("f16cdc5f-6366-5552-9ecd-c36290ab3d11"), + ad_login=None, + ad_domain=None, + username=None, + is_chat_admin=True, + is_chat_creator=True, + device=UserDevice( + manufacturer=None, + device_name=None, + os=None, + pushes=None, + timezone=None, + permissions=None, + platform=None, + platform_package_id=None, + app_version=None, + locale="en", + ), + ), + chat=Chat( + id=UUID("30dc1980-643a-00ad-37fc-7cc10d74e935"), + type=ChatTypes.PERSONAL_CHAT, + ), + raw_command=None, + ) + + +async def test__async_execute_raw_bot_command__maximum_filled_incoming_message( + datetime_formatter: Callable[[str], datetime], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "/hello", + "command_type": "user", + "data": {"message": "data"}, + "metadata": {"message": "metadata"}, + }, + "attachments": [], + "async_files": [ + { + "type": "image", + "file": "https://link.to/file", + "file_mime_type": "image/png", + "file_name": "pass.png", + "file_preview": "https://link.to/preview", + "file_preview_height": 300, + "file_preview_width": 300, + "file_size": 1502345, + "file_hash": "Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + "file_encryption_algo": "stream", + "chunk_size": 2097152, + "file_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", + }, + ], + "entities": [ + { + "type": "reply", + "data": { + "source_sync_id": "a7ffba12-8d0a-534e-8896-a0aa2d93a434", + "sender": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "body": "все равно документацию никто не читает...", + "mentions": [ + { + "mention_type": "contact", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "user_huid": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + "conn_type": "cts", + }, + }, + ], + "attachment": None, + "reply_type": "chat", + "source_group_chat_id": "918da23a-1c9a-506e-8a6f-1328f1499ee8", + "source_chat_name": "Serious Dev Chat", + }, + }, + { + "type": "forward", + "data": { + "group_chat_id": "918da23a-1c9a-506e-8a6f-1328f1499ee8", + "sender_huid": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "forward_type": "chat", + "source_chat_name": "Simple Chat", + "source_sync_id": "a7ffba12-8d0a-534e-8896-a0aa2d93a434", + "source_inserted_at": "2020-04-21T22:09:32.178Z", + }, + }, + { + "type": "mention", + "data": { + "mention_type": "contact", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "user_huid": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + "conn_type": "cts", + }, + }, + }, + ], + "source_sync_id": "bc3d06ed-7b2e-41ad-99f9-ca28adc2c88d", + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": "domain", + "ad_login": "login", + "app_version": "1.21.9", + "chat_type": "chat", + "device": "Firefox 91.0", + "device_meta": { + "permissions": { + "microphone": True, + "notifications": False, + }, + "pushes": False, + "timezone": "Europe/Moscow", + }, + "device_software": "Linux", + "group_chat_id": "30dc1980-643a-00ad-37fc-7cc10d74e935", + "host": "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": "Mozilla", + "platform": "web", + "platform_package_id": "ru.unlimitedtech.express", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + "username": "Ivanov Ivan Ivanovich", + }, + "proto_version": 4, + } + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert incoming_message == IncomingMessage( + bot=BotAccount( + id=UUID("24348246-6791-4ac0-9d86-b948cd6a0e46"), + host="cts.example.com", + ), + sync_id=UUID("6f40a492-4b5f-54f3-87ee-77126d825b51"), + source_sync_id=UUID("bc3d06ed-7b2e-41ad-99f9-ca28adc2c88d"), + body="/hello", + data={"message": "data"}, + metadata={"message": "metadata"}, + sender=UserSender( + huid=UUID("f16cdc5f-6366-5552-9ecd-c36290ab3d11"), + ad_login="login", + ad_domain="domain", + username="Ivanov Ivan Ivanovich", + is_chat_admin=True, + is_chat_creator=True, + device=UserDevice( + manufacturer="Mozilla", + device_name="Firefox 91.0", + os="Linux", + pushes=False, + timezone="Europe/Moscow", + permissions={"microphone": True, "notifications": False}, + platform=ClientPlatforms.WEB, + platform_package_id="ru.unlimitedtech.express", + app_version="1.21.9", + locale="en", + ), + ), + chat=Chat( + id=UUID("30dc1980-643a-00ad-37fc-7cc10d74e935"), + type=ChatTypes.PERSONAL_CHAT, + ), + raw_command=None, + file=Image( + type=AttachmentTypes.IMAGE, + filename="pass.png", + size=1502345, + is_async_file=True, + _file_id=UUID("8dada2c8-67a6-4434-9dec-570d244e78ee"), + _file_url="https://link.to/file", + _file_mimetype="image/png", + _file_hash="Jd9r+OKpw5y+FSCg1xNTSUkwEo4nCW1Sn1AkotkOpH0=", + ), + mentions=MentionList( + [ + Mention( + type=MentionTypes.CONTACT, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + ], + ), + forward=Forward( + chat_id=UUID("918da23a-1c9a-506e-8a6f-1328f1499ee8"), + author_id=UUID("c06a96fa-7881-0bb6-0e0b-0af72fe3683f"), + sync_id=UUID("a7ffba12-8d0a-534e-8896-a0aa2d93a434"), + ), + reply=Reply( + author_id=UUID("c06a96fa-7881-0bb6-0e0b-0af72fe3683f"), + sync_id=UUID("a7ffba12-8d0a-534e-8896-a0aa2d93a434"), + body="все равно документацию никто не читает...", + mentions=MentionList( + [ + Mention( + type=MentionTypes.CONTACT, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + ], + ), + ), + ) + + +async def test__async_execute_raw_bot_command__all_mention_types( + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + payload = { + "bot_id": "24348246-6791-4ac0-9d86-b948cd6a0e46", + "command": { + "body": "/hello", + "command_type": "user", + "data": {}, + "metadata": {}, + }, + "attachments": [], + "async_files": [], + "entities": [ + { + "type": "mention", + "data": { + "mention_type": "contact", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "user_huid": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + "conn_type": "cts", + }, + }, + }, + { + "type": "mention", + "data": { + "mention_type": "user", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "user_huid": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + "conn_type": "cts", + }, + }, + }, + { + "type": "mention", + "data": { + "mention_type": "channel", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "group_chat_id": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + }, + }, + }, + { + "type": "mention", + "data": { + "mention_type": "chat", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": { + "group_chat_id": "ab103983-6001-44e9-889e-d55feb295494", + "name": "Вася Иванов", + }, + }, + }, + { + "type": "mention", + "data": { + "mention_type": "all", + "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", + "mention_data": {}, + }, + }, + ], + "source_sync_id": None, + "sync_id": "6f40a492-4b5f-54f3-87ee-77126d825b51", + "from": { + "ad_domain": None, + "ad_login": None, + "app_version": None, + "chat_type": "chat", + "device": None, + "device_meta": { + "permissions": None, + "pushes": False, + "timezone": "Europe/Moscow", + }, + "device_software": None, + "group_chat_id": "30dc1980-643a-00ad-37fc-7cc10d74e935", + "host": "cts.example.com", + "is_admin": True, + "is_creator": True, + "locale": "en", + "manufacturer": None, + "platform": None, + "platform_package_id": None, + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + "username": None, + }, + "proto_version": 4, + } + + collector = HandlerCollector() + incoming_message: Optional[IncomingMessage] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + # Drop `raw_command` from asserting + incoming_message.raw_command = None + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert incoming_message + assert incoming_message.mentions == MentionList( + [ + Mention( + type=MentionTypes.CONTACT, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + Mention( + type=MentionTypes.USER, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + Mention( + type=MentionTypes.CHANNEL, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + Mention( + type=MentionTypes.CHAT, + entity_id=UUID("ab103983-6001-44e9-889e-d55feb295494"), + name="Вася Иванов", + ), + Mention( + type=MentionTypes.ALL, + entity_id=None, + name=None, + ), + ], + ) diff --git a/tests/test_lifespan.py b/tests/test_lifespan.py new file mode 100644 index 00000000..13197747 --- /dev/null +++ b/tests/test_lifespan.py @@ -0,0 +1,76 @@ +from http import HTTPStatus +from typing import Callable +from unittest.mock import Mock +from uuid import UUID + +import httpx +import pytest +from respx.router import MockRouter + +from botx import Bot, BotAccountWithSecret, HandlerCollector, IncomingMessage + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__shutdown__wait_for_active_handlers( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + bot.async_execute_bot_command(user_command) + await bot.shutdown() + + # - Assert - + correct_handler_trigger.assert_called_once() + + +async def test__startup__authorize_cant_get_token( + respx_mock: MockRouter, + loguru_caplog: pytest.LogCaptureFixture, + bot_account: BotAccountWithSecret, + host: str, + bot_id: UUID, + bot_signature: str, +) -> None: + # - Arrange - + token_endpoint = respx_mock.get( + f"https://{host}/api/v2/botx/bots/{bot_id}/token", + params={"signature": bot_signature}, + ).mock( + return_value=httpx.Response( + HTTPStatus.UNAUTHORIZED, + json={ + "status": "error", + }, + ), + ) + + collector = HandlerCollector() + + bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + await bot.startup() + + # - Assert - + assert token_endpoint.called + + assert "Can't get token for bot account: " in loguru_caplog.text + assert f"host - {host}, bot_id - {bot_id}" in loguru_caplog.text + + await bot.shutdown() diff --git a/tests/test_logs.py b/tests/test_logs.py new file mode 100644 index 00000000..56093595 --- /dev/null +++ b/tests/test_logs.py @@ -0,0 +1,128 @@ +from http import HTTPStatus +from typing import Any, Callable, Dict, Optional, cast +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + lifespan_wrapper, +) +from botx.models.attachments import AttachmentDocument, OutgoingAttachment + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__attachment__trimmed_in_incoming_message( + bot_account: BotAccountWithSecret, + api_incoming_message_factory: Callable[..., Dict[str, Any]], + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + payload = api_incoming_message_factory( + attachment={ + "data": { + "content": ( + "data:text/plain;base64," + "SGVsbG8sIGFtYXppbmcgd29ybGQhIFZlcnkgdmVyeSB2ZXJ5IHZlcnkgdm" + "VyeSB2ZXJ5IGxvbmcgdGV4dCB0byB0ZXN0IHRoYXQgdHJpbW1pbmcgY29u" + "dGVudCBkb2Vzbid0IGFmZmVjdCBmaWxlIGluIGluY29taW5nIG1lc3NhZ2U=" + ), + "file_name": "test_file.jpg", + }, + "type": "image", + }, + ) + collector = HandlerCollector() + file_data: Optional[bytes] = None + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal file_data + file = cast(AttachmentDocument, message.file) + file_data = file.content + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert "..." in loguru_caplog.text + assert file_data == ( + b"Hello, amazing world! Very very very very very very long text to" + b" test that trimming content doesn't affect file in incoming message" + ) + + +async def test__attachment__trimmed_in_outgoing_message( + respx_mock: MockRouter, + host: str, + bot_id: UUID, + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + endpoint = respx_mock.post( + f"https://{host}/api/v4/botx/notifications/direct", + headers={"Authorization": "Bearer token", "Content-Type": "application/json"}, + json={ + "group_chat_id": "054af49e-5e18-4dca-ad73-4f96b6de63fa", + "notification": {"status": "ok", "body": "Hi!"}, + "file": { + "file_name": "test.txt", + "data": ( + "data:text/plain;base64," + "SGVsbG8sIGFtYXppbmcgd29ybGQhIFZlcnkgdmVyeSB2ZXJ5IHZlcnkgdm" + "VyeSB2ZXJ5IGxvbmcgdGV4dCB0byB0ZXN0IHRoYXQgdHJpbW1pbmcgY29u" + "dGVudCBkb2Vzbid0IGFmZmVjdCBmaWxlIGluIGluY29taW5nIG1lc3NhZ2U=" + ), + }, + }, + ).mock( + return_value=httpx.Response( + HTTPStatus.ACCEPTED, + json={ + "status": "ok", + "result": {"sync_id": "21a9ec9e-f21f-4406-ac44-1a78d2ccf9e3"}, + }, + ), + ) + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + async with NamedTemporaryFile("wb+") as async_buffer: + await async_buffer.write( + b"Hello, amazing world! Very very very very very very long text to" + b" test that trimming content doesn't affect file in incoming message", + ) + await async_buffer.seek(0) + + file = await OutgoingAttachment.from_async_buffer( + async_buffer, + "test.txt", + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + await bot.send_message( + body="Hi!", + bot_id=bot_id, + chat_id=UUID("054af49e-5e18-4dca-ad73-4f96b6de63fa"), + file=file, + wait_callback=False, + ) + + # - Assert - + assert "..." in loguru_caplog.text + assert endpoint.called diff --git a/tests/test_middlewares.py b/tests/test_middlewares.py new file mode 100644 index 00000000..a7d29143 --- /dev/null +++ b/tests/test_middlewares.py @@ -0,0 +1,211 @@ +from typing import Callable +from unittest.mock import Mock + +import pytest + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + IncomingMessageHandlerFunc, + Middleware, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__middlewares__correct_order( + incoming_message_factory: Callable[..., IncomingMessage], + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + middlewares_called_order = [] + user_command = incoming_message_factory(body="/command") + + def middleware_factory(number: int) -> Middleware: + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + nonlocal middlewares_called_order + middlewares_called_order.append(number) + + await call_next(message, bot) + + return middleware + + collector = HandlerCollector( + middlewares=[middleware_factory(3), middleware_factory(4)], + ) + + @collector.command( + "/command", + description="My command", + middlewares=[middleware_factory(5), middleware_factory(6)], + ) + async def handler(message: IncomingMessage, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + middlewares=[middleware_factory(1), middleware_factory(2)], + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + correct_handler_trigger.assert_called_once() + + assert middlewares_called_order == [1, 2, 3, 4, 5, 6] + + +async def test__middlewares__called_in_default_handler( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + middlewares_called_order = [] + user_command = incoming_message_factory(body="/command") + + def middleware_factory(number: int) -> Middleware: + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + nonlocal middlewares_called_order + middlewares_called_order.append(number) + + await call_next(message, bot) + + return middleware + + collector = HandlerCollector( + middlewares=[middleware_factory(3), middleware_factory(4)], + ) + + @collector.default_message_handler( + middlewares=[middleware_factory(5), middleware_factory(6)], + ) + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + pass + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + middlewares=[middleware_factory(1), middleware_factory(2)], + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert middlewares_called_order == [1, 2, 3, 4, 5, 6] + + +async def test__middlewares__correct_child_collector_middlewares( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + middlewares_called_order = [] + user_command = incoming_message_factory(body="/command") + + def middleware_factory(number: int) -> Middleware: + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + nonlocal middlewares_called_order + middlewares_called_order.append(number) + + await call_next(message, bot) + + return middleware + + collector_1 = HandlerCollector( + middlewares=[middleware_factory(1), middleware_factory(2)], + ) + + @collector_1.command("/other-command", description="My command") + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + collector_2 = HandlerCollector( + middlewares=[middleware_factory(3), middleware_factory(4)], + ) + + @collector_2.command("/command", description="My command") + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + collector_1.include(collector_2) + built_bot = Bot(collectors=[collector_1], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert middlewares_called_order == [1, 2, 3, 4] + + +async def test__middlewares__correct_parent_collector_middlewares( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + middlewares_called_order = [] + user_command = incoming_message_factory(body="/command") + + def middleware_factory(number: int) -> Middleware: + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + nonlocal middlewares_called_order + middlewares_called_order.append(number) + + await call_next(message, bot) + + return middleware + + collector_1 = HandlerCollector( + middlewares=[middleware_factory(1), middleware_factory(2)], + ) + + @collector_1.command("/command", description="My command") + async def handler_1(message: IncomingMessage, bot: Bot) -> None: + pass + + collector_2 = HandlerCollector( + middlewares=[middleware_factory(3), middleware_factory(4)], + ) + + @collector_2.command("/other-command", description="My command") + async def handler_2(message: IncomingMessage, bot: Bot) -> None: + pass + + collector_1.include(collector_2) + built_bot = Bot(collectors=[collector_1], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert middlewares_called_order == [1, 2] diff --git a/tests/test_middlewares/__init__.py b/tests/test_middlewares/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_middlewares/test_authorization_middleware.py b/tests/test_middlewares/test_authorization_middleware.py deleted file mode 100644 index 32abdb96..00000000 --- a/tests/test_middlewares/test_authorization_middleware.py +++ /dev/null @@ -1,26 +0,0 @@ -import pytest - -from botx.middlewares.authorization import AuthorizationMiddleware - -pytestmark = pytest.mark.asyncio - - -async def test_obtaining_token_if_not_set(client, incoming_message): - bot = client.bot - bot.add_middleware(AuthorizationMiddleware) - bot_account = bot.get_account_by_bot_id(incoming_message.bot_id) - bot_account.token = None - - await client.send_command(incoming_message) - - assert bot.get_token_for_bot(incoming_message.bot_id) - - -async def test_doing_nothing_if_token_present(client, incoming_message): - bot = client.bot - bot.add_middleware(AuthorizationMiddleware) - token = bot.get_token_for_bot(incoming_message.bot_id) - - await client.send_command(incoming_message) - - assert bot.get_token_for_bot(incoming_message.bot_id) == token diff --git a/tests/test_middlewares/test_concurrency/__init__.py b/tests/test_middlewares/test_concurrency/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_middlewares/test_concurrency/fixtures.py b/tests/test_middlewares/test_concurrency/fixtures.py deleted file mode 100644 index 187d1cb1..00000000 --- a/tests/test_middlewares/test_concurrency/fixtures.py +++ /dev/null @@ -1,37 +0,0 @@ -import threading - -import pytest - -from botx import Message -from botx.middlewares.base import BaseMiddleware -from botx.typing import AsyncExecutor, Executor, SyncExecutor - - -class SyncMiddleware(BaseMiddleware): - def __init__(self, executor: Executor) -> None: - super().__init__(executor) - self.event = threading.Event() - - def dispatch(self, message: Message, call_next: SyncExecutor) -> None: - self.event.set() - call_next(message) - - -class AsyncMiddleware(BaseMiddleware): - def __init__(self, executor: Executor) -> None: - super().__init__(executor) - self.event = threading.Event() - - async def dispatch(self, message: Message, call_next: AsyncExecutor) -> None: - self.event.set() - await call_next(message) - - -@pytest.fixture() -def sync_middleware_class(): - return SyncMiddleware - - -@pytest.fixture() -def async_middleware_class(): - return AsyncMiddleware diff --git a/tests/test_middlewares/test_concurrency/test_async_processing.py b/tests/test_middlewares/test_concurrency/test_async_processing.py deleted file mode 100644 index 760e70a0..00000000 --- a/tests/test_middlewares/test_concurrency/test_async_processing.py +++ /dev/null @@ -1,28 +0,0 @@ -import threading - -import pytest - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_middlewares.test_concurrency.fixtures",) - - -async def test_async_middleware_receives_async_executor( - bot, - client, - incoming_message, - build_handler, - async_middleware_class, -): - bot.add_middleware(async_middleware_class) - - event = threading.Event() - bot.default(build_handler(event)) - - await client.send_command(incoming_message) - - assert event.is_set() - - executor = bot.exception_middleware.executor - - assert isinstance(executor, async_middleware_class) - assert executor.event.is_set() diff --git a/tests/test_middlewares/test_concurrency/test_sync_async_combinaton.py b/tests/test_middlewares/test_concurrency/test_sync_async_combinaton.py deleted file mode 100644 index 254b871d..00000000 --- a/tests/test_middlewares/test_concurrency/test_sync_async_combinaton.py +++ /dev/null @@ -1,44 +0,0 @@ -import threading - -import pytest - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_middlewares.test_concurrency.fixtures",) - - -@pytest.mark.parametrize( - "order", - [("sync", "sync"), ("sync", "async"), ("async", "sync"), ("async", "async")], -) -async def test_that_both_async_and_sync_middlewares_will_work( - bot, - client, - incoming_message, - async_middleware_class, - sync_middleware_class, - build_handler, - order, -): - event = threading.Event() - - if order[0] == "sync": - bot.add_middleware(sync_middleware_class) - else: - bot.add_middleware(async_middleware_class) - - if order[1] == "sync": - bot.add_middleware(sync_middleware_class) - else: - bot.add_middleware(async_middleware_class) - - bot.default(build_handler(event)) - - await client.send_command(incoming_message) - - assert event.is_set() - - middleware1 = bot.exception_middleware.executor - assert middleware1.event.is_set() - - middleware2 = middleware1.executor - assert middleware2.event.is_set() diff --git a/tests/test_middlewares/test_concurrency/test_sync_processing.py b/tests/test_middlewares/test_concurrency/test_sync_processing.py deleted file mode 100644 index f2bd7fc2..00000000 --- a/tests/test_middlewares/test_concurrency/test_sync_processing.py +++ /dev/null @@ -1,28 +0,0 @@ -import threading - -import pytest - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_middlewares.test_concurrency.fixtures",) - - -async def test_sync_middleware_receives_sync_executor( - bot, - client, - incoming_message, - build_handler, - sync_middleware_class, -): - bot.add_middleware(sync_middleware_class) - - event = threading.Event() - bot.default(build_handler(event)) - - await client.send_command(incoming_message) - - assert event.is_set() - - executor = bot.exception_middleware.executor - - assert isinstance(executor, sync_middleware_class) - assert executor.event.is_set() diff --git a/tests/test_middlewares/test_exception_middleware.py b/tests/test_middlewares/test_exception_middleware.py deleted file mode 100644 index eeb92185..00000000 --- a/tests/test_middlewares/test_exception_middleware.py +++ /dev/null @@ -1,101 +0,0 @@ -import threading - -import pytest - -from botx import TestClient - -pytestmark = pytest.mark.asyncio - - -async def test_handling_exception_with_custom_catcher( - bot, - incoming_message, - client, - build_exception_catcher, - build_failed_handler, - storage, -): - exc_for_raising = Exception("exception from handler") - - cather_event = threading.Event() - handler_event = threading.Event() - - bot.add_exception_handler(Exception, build_exception_catcher(cather_event)) - bot.default(build_failed_handler(exc_for_raising, handler_event)) - - await client.send_command(incoming_message) - - assert cather_event.is_set() - assert handler_event.is_set() - assert storage.exception == exc_for_raising - assert storage.message.incoming_message == incoming_message - - -async def test_handling_from_nearest_mro_handler( - bot, - incoming_message, - client, - build_exception_catcher, - build_failed_handler, - storage, -): - exc_for_raising = UnicodeError("exception from handler") - - exception_catcher_event = threading.Event() - value_error_catcher_event = threading.Event() - handler_event = threading.Event() - - bot.add_exception_handler( - Exception, - build_exception_catcher(exception_catcher_event), - ) - bot.add_exception_handler( - Exception, - build_exception_catcher(value_error_catcher_event), - ) - bot.default(handler=build_failed_handler(exc_for_raising, handler_event)) - - await client.send_command(incoming_message) - - assert not exception_catcher_event.is_set() - assert value_error_catcher_event.is_set() - assert handler_event.is_set() - assert storage.exception == exc_for_raising - assert storage.message.incoming_message == incoming_message - - -async def test_logging_exception_if_was_not_found( - bot, - incoming_message, - loguru_caplog, - build_failed_handler, -) -> None: - event = threading.Event() - bot.default(build_failed_handler(ValueError, event)) - - with TestClient(bot, suppress_errors=True) as client: - await client.send_command(incoming_message) - - assert event.is_set() - assert "uncaught ValueError exception" in loguru_caplog.text - - -async def test_logging_from_failed_exception_handler( - bot, - incoming_message, - client, - loguru_caplog, - build_exception_catcher, - build_failed_handler, -) -> None: - exc_for_raising = ValueError("exception from handler") - - handler_event = threading.Event() - - bot.add_exception_handler(Exception, build_exception_catcher(None)) - bot.default(build_failed_handler(exc_for_raising, handler_event)) - - await client.send_command(incoming_message) - - assert "ValueError('exception from handler')" in loguru_caplog.text - assert "uncaught AttributeError exception in error handler:" in loguru_caplog.text diff --git a/tests/test_middlewares/test_ns_middleware/__init__.py b/tests/test_middlewares/test_ns_middleware/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_middlewares/test_ns_middleware/fixtures.py b/tests/test_middlewares/test_ns_middleware/fixtures.py deleted file mode 100644 index 796cf204..00000000 --- a/tests/test_middlewares/test_ns_middleware/fixtures.py +++ /dev/null @@ -1,20 +0,0 @@ -import threading -from typing import Optional - -import pytest - -from botx import Message -from botx.middlewares.ns import register_next_step_handler - - -@pytest.fixture() -def build_handler_to_start_chain(): - def factory(next_handler_name: Optional[str], event: threading.Event): - def decorator(message: Message): - event.set() - if next_handler_name is not None: - register_next_step_handler(message, next_handler_name) - - return decorator - - return factory diff --git a/tests/test_middlewares/test_ns_middleware/test_arguments.py b/tests/test_middlewares/test_ns_middleware/test_arguments.py deleted file mode 100644 index 12a36b73..00000000 --- a/tests/test_middlewares/test_ns_middleware/test_arguments.py +++ /dev/null @@ -1,73 +0,0 @@ -import threading -from typing import Any - -import pytest - -from botx import Message -from botx.middlewares.ns import NextStepMiddleware, register_next_step_handler - -pytestmark = pytest.mark.asyncio - - -@pytest.fixture() -def build_handler_to_store_arguments(): - def factory(next_handler_name: str, event: threading.Event, **ns_args: Any): - def decorator(message: Message): - event.set() - register_next_step_handler(message, next_handler_name, **ns_args) - - return decorator - - return factory - - -@pytest.fixture() -def build_handler_to_save_message_in_storage(storage): - def factory(event: threading.Event): - def decorator(message: Message): - event.set() - storage.state = message.state - - return decorator - - return factory - - -async def test_setting_args_into_message_state( - bot, - incoming_message, - client, - build_handler_to_store_arguments, - build_handler_to_save_message_in_storage, - storage, -): - event1 = threading.Event() - event2 = threading.Event() - - bot.default( - handler=build_handler_to_store_arguments( - "ns_handler", - event1, - arg1=1, - arg2="2", - arg3=True, - ), - ) - - ns_handler = build_handler_to_save_message_in_storage(event2) - - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={"ns_handler": ns_handler}, - ) - - await client.send_command(incoming_message) - assert event1.is_set() - - await client.send_command(incoming_message) - assert event2.is_set() - - assert storage.state.arg1 == 1 - assert storage.state.arg2 == "2" - assert storage.state.arg3 diff --git a/tests/test_middlewares/test_ns_middleware/test_errors.py b/tests/test_middlewares/test_ns_middleware/test_errors.py deleted file mode 100644 index f876ebfb..00000000 --- a/tests/test_middlewares/test_ns_middleware/test_errors.py +++ /dev/null @@ -1,49 +0,0 @@ -import threading - -import pytest - -from botx.middlewares.ns import NextStepMiddleware - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_middlewares.test_ns_middleware.fixtures",) - - -async def test_error_for_ns_for_unregistered_handler( - bot, - client, - incoming_message, - build_handler_to_start_chain, -): - event = threading.Event() - - bot.default(handler=build_handler_to_start_chain("unknown_handler_name", event)) - - bot.add_middleware(NextStepMiddleware, bot=bot, functions={}) - - with pytest.raises(ValueError): - await client.send_command(incoming_message) - - assert event.is_set() - - -async def test_error_for_message_without_huid( - bot, - incoming_message, - client, - chat_created_message, - build_handler_to_start_chain, - build_handler_for_collector, -): - event = threading.Event() - bot.chat_created(build_handler_to_start_chain("ns_handler", event)) - - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={build_handler_for_collector("ns_handler")}, - ) - - with pytest.raises(ValueError): - await client.send_command(chat_created_message) - - assert event.is_set() diff --git a/tests/test_middlewares/test_ns_middleware/test_execution.py b/tests/test_middlewares/test_ns_middleware/test_execution.py deleted file mode 100644 index 2ef5193c..00000000 --- a/tests/test_middlewares/test_ns_middleware/test_execution.py +++ /dev/null @@ -1,80 +0,0 @@ -import threading - -import pytest - -from botx.middlewares.ns import NextStepMiddleware - -pytestmark = pytest.mark.asyncio -pytest_plugins = ("tests.test_middlewares.test_ns_middleware.fixtures",) - - -async def test_executing_ns_handlers( - bot, - incoming_message, - client, - build_handler_to_start_chain, -) -> None: - chain_start_event = threading.Event() - ns_handler_event = threading.Event() - - bot.default(handler=build_handler_to_start_chain("ns_handler", chain_start_event)) - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={"ns_handler": build_handler_to_start_chain(None, ns_handler_event)}, - ) - - incoming_message.command.body = "/start-ns" - - await client.send_command(incoming_message) - await client.send_command(incoming_message) - - assert chain_start_event.is_set() - assert ns_handler_event.is_set() - - -async def test_breaking_chain( - bot, - incoming_message, - client, - build_handler_to_start_chain, -): - break_handler_event = threading.Event() - chain_start_event = threading.Event() - ns_handler_event = threading.Event() - - chain_start_handler = build_handler_to_start_chain("ns_handler", chain_start_event) - - bot.handler( - handler=build_handler_to_start_chain(None, break_handler_event), - command="/break", - name="break_handler", - ) - bot.handler(handler=chain_start_handler, command="/ns-start", name="chain_start") - - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={ - "ns_handler": build_handler_to_start_chain("chain_start", ns_handler_event), - "chain_start": chain_start_handler, - }, - break_handler="break_handler", - ) - - incoming_message.command.body = "/ns-start" - await client.send_command(incoming_message) - assert chain_start_event.is_set() - chain_start_event.clear() - - await client.send_command(incoming_message) - assert ns_handler_event.is_set() - ns_handler_event.clear() - - await client.send_command(incoming_message) - assert chain_start_event.is_set() - chain_start_event.clear() - - incoming_message.command.body = "/break-handler" - await client.send_command(incoming_message) - assert break_handler_event.is_set() diff --git a/tests/test_middlewares/test_ns_middleware/test_registration.py b/tests/test_middlewares/test_ns_middleware/test_registration.py deleted file mode 100644 index 54c4dc5e..00000000 --- a/tests/test_middlewares/test_ns_middleware/test_registration.py +++ /dev/null @@ -1,72 +0,0 @@ -import pytest - -from botx.middlewares.ns import NextStepMiddleware, register_function_as_ns_handler - - -def test_register_middleware_with_functions_dict(bot, build_handler_for_collector): - functions = {"ns_handler": build_handler_for_collector("ns_handler")} - - bot.add_middleware(NextStepMiddleware, bot=bot, functions=functions) - - assert [bot.state.ns_collector.handler_for(name) for name in functions] - - -def test_register_ns_middleware_using_functions_set(bot, build_handler_for_collector): - functions = {build_handler_for_collector("ns_handler")} - - bot.add_middleware(NextStepMiddleware, bot=bot, functions=functions) - - assert [bot.state.ns_collector.handler_for(name) for name in ["ns_handler"]] - - -def test_no_duplicate_handlers_registration(bot, build_handler_for_collector): - bot.add_middleware(NextStepMiddleware, bot=bot, functions={}) - - handler = build_handler_for_collector("ns_handler") - - register_function_as_ns_handler(bot, handler) - - with pytest.raises(ValueError): - register_function_as_ns_handler(bot, handler) - - -def test_register_break_handler_as_string(bot, build_handler_for_collector): - bot.handler(handler=build_handler_for_collector("break_handler"), command="/break") - - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={}, - break_handler="break_handler", - ) - - assert bot.state.ns_collector.handler_for("break_handler") == bot.handler_for( - "break_handler", - ) - - -def test_register_break_handler_as_handler(bot, build_handler_for_collector): - bot.handler(build_handler_for_collector("break_handler"), command="/break") - - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={}, - break_handler=bot.handler_for("break_handler"), - ) - - assert bot.state.ns_collector.handler_for("break_handler") == bot.handler_for( - "break_handler", - ) - - -def test_register_break_handler_as_function(bot, build_handler_for_collector): - handler = build_handler_for_collector("break_handler") - bot.add_middleware( - NextStepMiddleware, - bot=bot, - functions={}, - break_handler=handler, - ) - - assert bot.state.ns_collector.handler_for("break_handler").handler == handler diff --git a/tests/test_missing.py b/tests/test_missing.py new file mode 100644 index 00000000..25ed6b4b --- /dev/null +++ b/tests/test_missing.py @@ -0,0 +1,10 @@ +import pytest + +from botx.missing import Undefined, not_undefined + + +def test__not_undefined__all_args_undefined() -> None: + with pytest.raises(ValueError) as exc: + not_undefined(Undefined, Undefined) + + assert "All arguments" in str(exc.value) diff --git a/tests/test_models/__init__.py b/tests/test_models/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_models/test_attachments.py b/tests/test_models/test_attachments.py deleted file mode 100644 index 051d09a7..00000000 --- a/tests/test_models/test_attachments.py +++ /dev/null @@ -1,229 +0,0 @@ -import pytest - -from botx import MessageBuilder -from botx.models.attachments import Video, Voice - - -def test_is_link_in_attachment(): - builder = MessageBuilder() - builder.link() - assert builder.message.attachments.__root__[0].data.is_link() - - -def test_is_mail_in_attachment(): - builder = MessageBuilder() - mailto_url = "mailto:mail@mail.com" - builder.link(url=mailto_url) - assert builder.message.attachments.__root__[0].data.is_mail() - - -def test_is_telephone_number_in_attachment(): - builder = MessageBuilder() - tel_url = "tel://+77777777777" - builder.link(url=tel_url) - assert builder.message.attachments.__root__[0].data.is_telephone() - - -def test_mailto_property_in_attachment(): - builder = MessageBuilder() - mailto_url = "mailto:mail@mail.com" - builder.link(mailto_url) - assert builder.message.attachments.__root__[0].data.mailto == "mail@mail.com" - - -def test_raising_missing_mailto(): - builder = MessageBuilder() - builder.link() - with pytest.raises(AttributeError): - builder.message.attachments.__root__[0].data.mailto - - -def test_tel_property_in_attachment(): - builder = MessageBuilder() - tel_url = "tel://+77777777777" - builder.link(url=tel_url) - assert builder.message.attachments.__root__[0].data.tel == "+77777777777" - - -def test_raising_missing_tel(): - builder = MessageBuilder() - builder.link() - with pytest.raises(AttributeError): - builder.message.attachments.__root__[0].data.tel - - -def test_image_in_attachments(): - builder = MessageBuilder() - builder.document() - builder.image() - assert builder.message.attachments.image - - -def test_missing_image_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.image - - -def test_document_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.document() - assert builder.message.attachments.document - - -def test_missing_document_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.document - - -def test_location_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.location() - assert builder.message.attachments.location - - -def test_missing_location_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.location - - -def test_contact_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.contact() - assert builder.message.attachments.contact - - -def test_missing_contact_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.contact - - -def test_voice_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.voice() - assert builder.message.attachments.voice - - -def test_missing_voice_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.voice - - -def test_video_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.video() - assert builder.message.attachments.video - - -def test_missing_video_in_attachments(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.video - - -def test_link_in_attachments(): - builder = MessageBuilder() - builder.image() - builder.link() - assert builder.message.attachments.link - - -def test_missing_link_in_attachments(): - builder = MessageBuilder() - builder.link(url="mailto:mail@mail.com") - with pytest.raises(AttributeError): - builder.message.attachments.link - - -def test_email_in_attachments(): - builder = MessageBuilder() - mailto_url = "mailto:mail@mail.com" - builder.image() - builder.link(url=mailto_url) - assert builder.message.attachments.email == "mail@mail.com" - - -def test_missing_email_in_attachments(): - builder = MessageBuilder() - builder.link(url="https://any.com") - with pytest.raises(AttributeError): - builder.message.attachments.email - - -def test_telephone_in_attachments(): - builder = MessageBuilder() - tel_url = "tel://+77777777777" - builder.image() - builder.link(url=tel_url) - assert builder.message.attachments.telephone == "+77777777777" - - -def test_missing_telephone_in_attachments(): - builder = MessageBuilder() - builder.link(url="mailto:mail@mail.com") - with pytest.raises(AttributeError): - builder.message.attachments.telephone - - -@pytest.mark.parametrize( - "attach", - [lambda x: x.document, lambda x: x.image, lambda x: x.video], -) -def test_file_in_attachments(attach): - builder = MessageBuilder() - attach(builder)() - assert builder.message.attachments.file - - -def test_no_file_in_message(): - builder = MessageBuilder() - builder.link() - with pytest.raises(AttributeError): - builder.message.attachments.file - - -def test_file_with_unsupported_extension(): - builder = MessageBuilder() - builder.document(file_name="test.py") - assert builder.message.attachments.file - - -@pytest.mark.parametrize("len_of_attachments", [1, 2, 3]) -def test_get_all_attachments(len_of_attachments): - builder = MessageBuilder() - for _ in range(len_of_attachments): - builder.document() - assert len(builder.message.attachments.all_attachments) == len_of_attachments - - -def test_video_attach_has_video_type(): - builder = MessageBuilder() - builder.video() - assert isinstance(builder.message.attachments.video, Video) - - -def test_voice_attach_has_voice_type(): - builder = MessageBuilder() - builder.voice() - assert isinstance(builder.message.attachments.voice, Voice) - - -def test_no_attach_type(): - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.attachments.attach_type - - -def test_attach_type(): - builder = MessageBuilder() - builder.link() - builder.message.attachments.attach_type == "link" diff --git a/tests/test_models/test_buttons.py b/tests/test_models/test_buttons.py deleted file mode 100644 index 0e8dc644..00000000 --- a/tests/test_models/test_buttons.py +++ /dev/null @@ -1,27 +0,0 @@ -import pytest -from pydantic import ValidationError - -from botx.models.buttons import Button, ButtonOptions - - -class CustomButton(Button): - """Button without custom behaviour.""" - - -def test_label_will_be_set_to_command_if_none(): - assert CustomButton(command="/cmd").label == "/cmd" - - -def test_label_can_be_set_if_passed_explicitly(): - assert CustomButton(command="/cmd", label="temp").label == "temp" - - -def test_empty_label(): - assert CustomButton(command="/cmd", label="").label == "" - - -def test_create_button_options_with_invalid_hsize(): - with pytest.raises(ValidationError) as exc_info: - ButtonOptions(h_size=0) - - assert "should be positive integer" in str(exc_info) diff --git a/tests/test_models/test_credentials.py b/tests/test_models/test_credentials.py deleted file mode 100644 index f9b02c41..00000000 --- a/tests/test_models/test_credentials.py +++ /dev/null @@ -1,48 +0,0 @@ -from uuid import UUID - -import pytest - -from botx import BotXCredentials -from botx.clients.methods.v2.bots.token import Token - - -def test_calculating_signature_for_token(host) -> None: - bot_id = UUID("8dada2c8-67a6-4434-9dec-570d244e78ee") - account = BotXCredentials( - host=host, - secret_key="secret", - bot_id=bot_id, - ) - signature = "904E39D3BC549C71F4A4BDA66AFCDA6FC90D471A64889B45CC8D2288E56526AD" - assert account.signature == signature - - -@pytest.mark.asyncio() -async def test_auth_to_each_known_account(bot, client) -> None: - accounts_len = len(bot.bot_accounts) - for account in bot.bot_accounts: - account.token = None - - await bot.authorize() - assert len(client.requests) == accounts_len - - for request in client.requests: - assert isinstance(request, Token) - - for account in bot.bot_accounts: - assert account.token is not None - - -@pytest.mark.asyncio() -async def test_auth_with_wrong_credentials(bot, host) -> None: - bot_id = UUID("8dada2c8-67a6-4434-9dec-570d244e78ee") - bot.bot_accounts = [ - BotXCredentials( - host=host, - secret_key="wrong_secret", - bot_id=bot_id, - ), - ] - await bot.authorize() - - assert bot.bot_accounts[0].token is None diff --git a/tests/test_models/test_datastructures.py b/tests/test_models/test_datastructures.py deleted file mode 100644 index aaa3807c..00000000 --- a/tests/test_models/test_datastructures.py +++ /dev/null @@ -1,20 +0,0 @@ -import pytest - -from botx.models.datastructures import State - - -def test_passed_state_applied(): - state = State({"arg": 1}) - assert state.arg == 1 - - -def test_state_can_be_set(): - state = State() - state.arg = 1 - assert state.arg == 1 - - -def test_state_will_raise_error_on_empty_attribute(): - state = State() - with pytest.raises(AttributeError): - _ = state.arg # noqa: WPS122 diff --git a/tests/test_models/test_entities.py b/tests/test_models/test_entities.py deleted file mode 100644 index 2867dd37..00000000 --- a/tests/test_models/test_entities.py +++ /dev/null @@ -1,150 +0,0 @@ -import uuid - -import pytest -from pydantic import ValidationError - -from botx import ( - ChatMention, - Mention, - MentionTypes, - Message, - MessageBuilder, - UserMention, -) - - -class TestMentions: - @pytest.mark.parametrize("mention_id", [None, uuid.uuid4()]) - def test_mention_id_will_be_generated_if_missed(self, mention_id): - mention = Mention( - mention_id=mention_id, - mention_data=UserMention(user_huid=uuid.uuid4()), - ) - assert mention.mention_id is not None - - def test_error_when_no_mention_data(self): - with pytest.raises(ValidationError): - Mention(mention_type=MentionTypes.user) - - @pytest.mark.parametrize( - ("mention_data", "mention_type"), - [ - (UserMention(user_huid=uuid.uuid4()), MentionTypes.user), - (UserMention(user_huid=uuid.uuid4()), MentionTypes.contact), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.chat), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.channel), - ], - ) - def test_mention_corresponds_data_by_type(self, mention_data, mention_type) -> None: - mention = Mention(mention_data=mention_data, mention_type=mention_type) - assert mention.mention_type == mention_type - - @pytest.mark.parametrize( - ("mention_data", "mention_type"), - [ - (UserMention(user_huid=uuid.uuid4()), MentionTypes.chat), - (UserMention(user_huid=uuid.uuid4()), MentionTypes.channel), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.user), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.contact), - ], - ) - def test_error_when_data_not_corresponds_type( - self, - mention_data, - mention_type, - ) -> None: - with pytest.raises(ValidationError): - assert Mention(mention_data=mention_data, mention_type=mention_type) - - @pytest.mark.parametrize( - ("mention_data", "mention_type"), - [ - (UserMention(user_huid=uuid.uuid4()), MentionTypes.user), - (UserMention(user_huid=uuid.uuid4()), MentionTypes.contact), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.chat), - (ChatMention(group_chat_id=uuid.uuid4()), MentionTypes.channel), - ], - ) - def test_mention_in_message(self, mention_data, mention_type) -> None: - builder = MessageBuilder() - builder.mention( - mention=Mention(mention_data=mention_data, mention_type=mention_type), - ) - assert builder.message.entities.mentions[0] - - def test_mention_not_in_message(self, bot) -> None: - builder = MessageBuilder() - message = Message.from_dict(builder.message.dict(), bot) - assert message.entities.mentions == [] - - def test_user_mention_botx_format(self) -> None: - mention_id = uuid.uuid4() - user_mention = Mention( - mention_id=mention_id, - mention_data=UserMention(user_huid=uuid.uuid4()), - mention_type=MentionTypes.user, - ) - - formatted_mention = user_mention.to_botx_format() - - assert formatted_mention == f"@{{mention:{mention_id}}}" - - def test_contact_mention_botx_format(self) -> None: - mention_id = uuid.uuid4() - contact_mention = Mention( - mention_id=mention_id, - mention_data=UserMention(user_huid=uuid.uuid4()), - mention_type=MentionTypes.contact, - ) - - formatted_mention = contact_mention.to_botx_format() - - assert formatted_mention == f"@@{{mention:{mention_id}}}" - - def test_chat_mention_botx_format(self) -> None: - mention_id = uuid.uuid4() - chat_mention = Mention( - mention_id=mention_id, - mention_data=ChatMention(group_chat_id=uuid.uuid4()), - mention_type=MentionTypes.chat, - ) - - formatted_mention = chat_mention.to_botx_format() - - assert formatted_mention == f"##{{mention:{mention_id}}}" - - def test_channel_mention_botx_format(self) -> None: - mention_id = uuid.uuid4() - channel_mention = Mention( - mention_id=mention_id, - mention_data=ChatMention(group_chat_id=uuid.uuid4()), - mention_type=MentionTypes.channel, - ) - - formatted_mention = channel_mention.to_botx_format() - - assert formatted_mention == f"##{{mention:{mention_id}}}" - - -class TestReply: - def test_reply_in_message(self, message) -> None: - builder = MessageBuilder() - builder.reply(message=message) - assert builder.message.entities.reply.source_sync_id == message.sync_id - - def test_reply_not_in_message(self) -> None: - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.entities.reply - - -class TestForward: - def test_forward_in_message(self, message) -> None: - builder = MessageBuilder() - builder.forward(message=message) - assert builder.message.entities.forward.source_sync_id == message.sync_id - - def test_forward_not_in_message(self) -> None: - builder = MessageBuilder() - with pytest.raises(AttributeError): - builder.message.entities.forward diff --git a/tests/test_models/test_errors.py b/tests/test_models/test_errors.py deleted file mode 100644 index cdadeaf7..00000000 --- a/tests/test_models/test_errors.py +++ /dev/null @@ -1,16 +0,0 @@ -import pytest -from pydantic import ValidationError - -from botx import BotDisabledErrorData, BotDisabledResponse - - -def test_error_for_missing_status_message_field(): - with pytest.raises(ValidationError): - BotDisabledResponse(error_data={}) - - -def test_doing_nothing_when_passed_error_data_model(): - response = BotDisabledResponse( - error_data=BotDisabledErrorData(status_message="test"), - ) - assert response.error_data.status_message == "test" diff --git a/tests/test_models/test_files/__init__.py b/tests/test_models/test_files/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_models/test_files/test_attributes.py b/tests/test_models/test_files/test_attributes.py deleted file mode 100644 index 43d8f984..00000000 --- a/tests/test_models/test_files/test_attributes.py +++ /dev/null @@ -1,25 +0,0 @@ -from botx import File - - -def test_retrieving_file_data_in_bytes(): - assert File.from_string(b"test", filename="test.txt").data_in_bytes == b"test" - - -def test_retrieving_file_data_in_base64(): - assert File.from_string(b"test", filename="test.txt").data_in_base64 == "dGVzdA==" - - -def test_retrieving_txt_media_type(): - assert File.from_string(b"test", filename="test.txt").media_type == "text/plain" - - -def test_retrieving_png_media_type(): - assert File.from_string(b"test", filename="test.png").media_type == "image/png" - - -def test_retrieving_file_size(): - assert File.from_string(b"file\ncontents", filename="test.txt").size_in_bytes == 13 - - -def test_get_ext_by_unsupported_mimetype(): - assert File.get_ext_by_mimetype("application/javascript") is None diff --git a/tests/test_models/test_files/test_constructing.py b/tests/test_models/test_files/test_constructing.py deleted file mode 100644 index 0b0a5908..00000000 --- a/tests/test_models/test_files/test_constructing.py +++ /dev/null @@ -1,85 +0,0 @@ -from io import BytesIO, StringIO - -import aiofiles -import pytest - -from botx import File - - -@pytest.mark.parametrize("extension", [".docx", ".txt", ".html", ".pdf"]) -def test_file_creation_with_right_extension(extension): - File(file_name=f"tmp{extension}", data="") - - -@pytest.mark.parametrize( - ("io_cls", "file_data", "file_name"), - [(StringIO, "test", "test.txt"), (BytesIO, b"test", "test.txt")], -) -@pytest.mark.parametrize("explicit_file_name", ["test2.txt", None]) -def test_creating_file_from_io_with_name( - io_cls, - file_data, - file_name, - explicit_file_name, -): - created_file = io_cls(file_data) - if not explicit_file_name: - created_file.name = file_name - - assert File.from_file(created_file, filename=explicit_file_name) == File( - file_name=explicit_file_name or file_name, - data="data:text/plain;base64,dGVzdA==", - ) - - -@pytest.mark.parametrize("file_data", ["test", b"test"]) -def test_creating_file_from_string(file_data): - assert File.from_string(file_data, filename="test.txt") == File( - file_name="test.txt", - data="data:text/plain;base64,dGVzdA==", - ) - - -@pytest.fixture() -def filename(): - return "test.txt" - - -@pytest.fixture() -def origin_data(): - return b"Hello,\nworld!" - - -@pytest.fixture() -def encoded_data(): - return "data:text/plain;base64,SGVsbG8sCndvcmxkIQ==" - - -@pytest.fixture() -def temp_file(tmp_path, filename, origin_data): - file_path = tmp_path / filename - file_path.write_bytes(origin_data) - - return file_path - - -@pytest.mark.asyncio() -async def test_async_from_file(temp_file, encoded_data): - async with aiofiles.open(temp_file, "rb") as fo: - file = await File.async_from_file(fo) - - assert file.file_name == temp_file.name - assert file.data == encoded_data - - -def test_file_chunks(filename, encoded_data, origin_data): - file = File.construct(file_name=filename, data=encoded_data) - temp_file = BytesIO() - - with file.file_chunks() as chunks: - for chunk in chunks: - temp_file.write(chunk) - - temp_file.seek(0) - - assert temp_file.read() == origin_data diff --git a/tests/test_models/test_messages/__init__.py b/tests/test_models/test_messages/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_models/test_messages/test_messages.py b/tests/test_models/test_messages/test_messages.py deleted file mode 100644 index 717e5928..00000000 --- a/tests/test_models/test_messages/test_messages.py +++ /dev/null @@ -1,663 +0,0 @@ -import uuid -from io import BytesIO, StringIO - -import pytest - -from botx import ( - Bot, - BubbleElement, - ChatMention, - File, - IncomingMessage, - KeyboardElement, - Mention, - MentionTypes, - Message, - MessageBuilder, - MessageMarkup, - MessageOptions, - NotificationOptions, - SendingCredentials, - SendingMessage, - UserMention, -) - - -@pytest.fixture() -def incoming_message() -> IncomingMessage: - return IncomingMessage.parse_obj( - { - "sync_id": "a465f0f3-1354-491c-8f11-f400164295cb", - "command": { - "body": "system:chat_created", - "command_type": "system", - "data": { - "group_chat_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", - "chat_type": "group_chat", - "name": "Meeting Room", - "creator": "ab103983-6001-44e9-889e-d55feb295494", - "members": [ - { - "huid": "ab103983-6001-44e9-889e-d55feb295494", - "name": "Bob", - "user_kind": "user", - "admin": True, - }, - { - "huid": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", - "name": "Funny Bot", - "user_kind": "botx", - "admin": False, - }, - ], - }, - "metadata": {"account_id": 94}, - }, - "file": None, - "from": { - "user_huid": None, - "group_chat_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", - "ad_login": None, - "ad_domain": None, - "username": None, - "chat_type": "group_chat", - "host": "cts.ccteam.ru", - "is_admin": False, - "is_creator": False, - }, - "bot_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", - "entities": [ - { - "type": "mention", - "data": { - "mention_type": "contact", - "mention_id": "c06a96fa-7881-0bb6-0e0b-0af72fe3683f", - "mention_data": { - "user_huid": "ab103983-6001-44e9-889e-d55feb295494", - "name": "User", - }, - }, - }, - ], - }, - ) - - -@pytest.fixture() -def embed_mention_with_name() -> str: - user_huid = uuid.uuid4() - return f"" - - -@pytest.fixture() -def embed_mention_without_name() -> str: - user_huid = uuid.uuid4() - return f"" - - -@pytest.fixture() -def embed_mention_with_wrong_type() -> str: - user_huid = uuid.uuid4() - return f"" - - -def test_setting_ui_flag_property_for_common_message() -> None: - msg = Message.from_dict( - { - "sync_id": "a465f0f3-1354-491c-8f11-f400164295cb", - "source_sync_id": "ff934be3-a03f-45d8-b315-738ba1ddec45", - "command": { - "body": "/cmd", - "command_type": "user", - "data": {"ui": True}, - "metadata": {"account_id": 94}, - }, - "file": None, - "from": { - "user_huid": None, - "group_chat_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", - "ad_login": None, - "ad_domain": None, - "username": None, - "chat_type": "group_chat", - "host": "cts.ccteam.ru", - "is_admin": False, - "is_creator": False, - }, - "bot_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", - }, - Bot(), - ) - - assert msg.source_sync_id == uuid.UUID("ff934be3-a03f-45d8-b315-738ba1ddec45") - - -def test_setting_ui_flag_property_for_system_message(incoming_message) -> None: - msg = Message.from_dict(incoming_message.dict(), Bot()) - assert not msg.source_sync_id - - -@pytest.fixture() -def sending_message() -> SendingMessage: - return SendingMessage( - text="text", - file=File.from_string(b"data", filename="file.txt"), - credentials=SendingCredentials( - sync_id=uuid.uuid4(), - bot_id=uuid.uuid4(), - host="host", - ), - markup=MessageMarkup( - bubbles=[[BubbleElement(command="")]], - keyboard=[[KeyboardElement(command="")]], - ), - options=MessageOptions( - recipients=[uuid.uuid4()], - mentions=[ - Mention(mention_data=UserMention(user_huid=uuid.uuid4())), - Mention( - mention_data=UserMention(user_huid=uuid.uuid4()), - mention_type=MentionTypes.contact, - ), - Mention( - mention_data=ChatMention(group_chat_id=uuid.uuid4()), - mention_type=MentionTypes.chat, - ), - ], - notifications=NotificationOptions(send=False, force_dnd=True), - ), - ) - - -def test_message_is_proxy_to_incoming_message(incoming_message) -> None: - msg = Message.from_dict(incoming_message.dict(), Bot()) - assert msg.sync_id == incoming_message.sync_id - assert msg.command == incoming_message.command - assert msg.file == incoming_message.file - assert msg.user == incoming_message.user - assert msg.bot_id == incoming_message.bot_id - assert msg.body == incoming_message.command.body - assert msg.data == {**msg.metadata, **incoming_message.command.data_dict} - assert msg.metadata == incoming_message.command.metadata - assert msg.user_huid == incoming_message.user.user_huid - assert msg.ad_login == incoming_message.user.ad_login - assert msg.group_chat_id == incoming_message.user.group_chat_id - assert msg.chat_type == incoming_message.user.chat_type - assert msg.host == incoming_message.user.host - assert msg.credentials.sync_id == incoming_message.sync_id - assert msg.credentials.bot_id == incoming_message.bot_id - assert msg.credentials.host == incoming_message.user.host - assert msg.entities == incoming_message.entities - assert msg.incoming_message == incoming_message - - -class TestBuildingSendingMessage: - def test_building_from_message(self, sending_message: SendingMessage) -> None: - builder = MessageBuilder() - msg = Message(message=builder.message, bot=Bot()) - sending_msg = SendingMessage.from_message( - text=sending_message.text, - message=msg, - ) - assert sending_msg.host == msg.host - assert sending_msg.sync_id == msg.sync_id - assert sending_msg.bot_id == msg.bot_id - - def test_building_with_embed_mentions( - self, - sending_message: SendingMessage, - ) -> None: - credentials = sending_message.credentials - user_huid = uuid.uuid4() - - mention_id = uuid.uuid4() - text = ( - f"Text with embed_mention: " - ) - - msg = SendingMessage(text=text, credentials=credentials, embed_mentions=True) - - assert msg.text == f"Text with embed_mention: @{{mention:{mention_id}}}" - assert msg.options.mentions - - class TestCredentialsBuilding: - def test_only_credentials_or_separate_credential_parts( - self, - sending_message: SendingMessage, - ) -> None: - with pytest.raises(AssertionError): - _ = SendingMessage( - sync_id=sending_message.sync_id, - bot_id=sending_message.bot_id, - host=sending_message.host, - credentials=sending_message.credentials, - ) - - def test_credentials_will_be_built_from_credential_parts( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - sync_id=sending_message.sync_id, - bot_id=sending_message.bot_id, - host=sending_message.host, - ) - assert msg.credentials == sending_message.credentials - - def test_merging_message_id_into_credentials( - self, - sending_message: SendingMessage, - ) -> None: - message_id = uuid.uuid4() - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - message_id=message_id, - ) - assert msg.credentials.message_id == message_id - - def test_leaving_credentials_message_id_into_credentials_if_was_set( - self, - sending_message: SendingMessage, - ) -> None: - message_id = uuid.uuid4() - sending_message.credentials.message_id = message_id - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - message_id=uuid.uuid4(), - ) - assert msg.credentials.message_id == sending_message.credentials.message_id - - class TestMarkupBuilding: - def test_markup_creation_from_bubbles( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - bubbles=sending_message.markup.bubbles, - ) - assert msg.markup.keyboard == [] - assert msg.markup.bubbles == sending_message.markup.bubbles - - def test_markup_creation_from_keyboard( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - keyboard=sending_message.markup.keyboard, - ) - assert msg.markup.keyboard == sending_message.markup.keyboard - assert msg.markup.bubbles == [] - - def test_markup_creation_from_bubbles_and_keyboard( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - bubbles=sending_message.markup.bubbles, - keyboard=sending_message.markup.keyboard, - ) - assert msg.markup == sending_message.markup - - def test_only_markup_or_separate_markup_parts( - self, - sending_message: SendingMessage, - ) -> None: - with pytest.raises(AssertionError): - _ = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - bubbles=sending_message.markup.bubbles, - keyboard=sending_message.markup.keyboard, - markup=sending_message.markup, - ) - - class TestOptionsBuilding: - def test_options_from_mentions(self, sending_message: SendingMessage) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - mentions=sending_message.options.mentions, - ) - assert msg.options.mentions == sending_message.options.mentions - - def test_options_from_recipients(self, sending_message: SendingMessage) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - recipients=sending_message.options.recipients, - ) - assert msg.options.recipients == sending_message.options.recipients - - def test_options_from_notification_options( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - notification_options=sending_message.options.notifications, - ) - assert msg.options.notifications == sending_message.options.notifications - - def test_option_from_message_options( - self, - sending_message: SendingMessage, - ) -> None: - msg = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - options=sending_message.options, - ) - assert msg.options == sending_message.options - - def test_only_options_or_separate_options_parts( - self, - sending_message: SendingMessage, - ) -> None: - with pytest.raises(AssertionError): - _ = SendingMessage( - text=sending_message.text, - credentials=sending_message.credentials, - options=sending_message.options, - mentions=sending_message.options.mentions, - recipients=sending_message.options.recipients, - notification_options=sending_message.options.notifications, - ) - - -class TestSendingMessageProperties: - def test_message_text(self, sending_message: SendingMessage) -> None: - sending_message.text = "test" - assert sending_message.text == "test" - - def test_metadata(self, sending_message: SendingMessage) -> None: - value = {"account_id", 94} - - sending_message.metadata = value - assert sending_message.metadata == value - - class TestMessageFile: - def test_message_file(self, sending_message: SendingMessage) -> None: - file = sending_message.file - sending_message.file = file - assert sending_message.file == file - - def test_message_file_from_file(self, sending_message: SendingMessage) -> None: - original_file = sending_message.file - sending_message.add_file(File.from_file(original_file.file)) - assert sending_message.file == original_file - - def test_message_file_from_string_file( - self, - sending_message: SendingMessage, - ) -> None: - original_file = sending_message.file - sending_message.add_file( - StringIO(original_file.file.read().decode()), - filename=original_file.file_name, - ) - assert sending_message.file == original_file - - def test_message_file_from_bytes_file( - self, - sending_message: SendingMessage, - ) -> None: - original_file = sending_message.file - sending_message.add_file( - BytesIO(original_file.file.read()), - filename=original_file.file_name, - ) - assert sending_message.file == original_file - - def test_message_markup(self, sending_message: SendingMessage) -> None: - markup = MessageMarkup(bubbles=[[BubbleElement(command="/test")]]) - sending_message.markup = markup - assert sending_message.markup == markup - - def test_message_options(self, sending_message: SendingMessage) -> None: - options = MessageOptions() - sending_message.options = options - assert sending_message.options == options - - def test_message_sync_id(self, sending_message: SendingMessage) -> None: - sync_id = uuid.uuid4() - sending_message.sync_id = sync_id - assert sending_message.sync_id == sync_id - - def test_message_chat_id(self, sending_message: SendingMessage) -> None: - chat_id = uuid.uuid4() - sending_message.chat_id = chat_id - assert sending_message.chat_id == chat_id - - def test_message_bot_id(self, sending_message: SendingMessage) -> None: - bot_id = uuid.uuid4() - sending_message.bot_id = bot_id - assert sending_message.bot_id == bot_id - - def test_message_host(self, sending_message: SendingMessage) -> None: - host = "example.com" - sending_message.host = host - assert sending_message.host == host - - class TestMentioning: - def test_mentioning_user(self, sending_message: SendingMessage) -> None: - sending_message.payload.options.mentions = [] - user_huid = uuid.uuid4() - user_name = "test" - sending_message.mention_user(user_huid, user_name) - mention = sending_message.payload.options.mentions[0] - assert mention.mention_type == MentionTypes.user - - def test_mentioning_contact(self, sending_message: SendingMessage) -> None: - sending_message.payload.options.mentions = [] - user_huid = uuid.uuid4() - user_name = "test" - sending_message.mention_contact(user_huid, user_name) - mention = sending_message.payload.options.mentions[0] - assert mention.mention_type == MentionTypes.contact - - def test_mentioning_chat(self, sending_message: SendingMessage) -> None: - sending_message.payload.options.mentions = [] - chat_id = uuid.uuid4() - chat_name = "test" - sending_message.mention_chat(chat_id, chat_name) - mention = sending_message.payload.options.mentions[0] - assert mention.mention_type == MentionTypes.chat - - def test_wrong_mention_chat(self, sending_message: SendingMessage) -> None: - wrong_mention_chat = { - "mention_type": MentionTypes.chat, - "mention_id": uuid.uuid4(), - "mention_data": {"foo": "bar"}, - } - with pytest.raises(ValueError): - Mention.parse_obj(wrong_mention_chat) - - def test_mention_data_error(self): - mention_all = { - "mention_type": "all", - "mention_id": uuid.uuid4(), - "mention_data": {}, - } - mention = Mention.parse_obj(mention_all) - assert mention.mention_data is None - - class TestBuildingMentions: - def test_build_embeddable_user_mention(self) -> None: - user_huid = uuid.uuid4() - - embeddable_mention = SendingMessage.build_embeddable_user_mention(user_huid) - - assert embeddable_mention.startswith(f" None: - user_huid = uuid.uuid4() - - embeddable_mention = SendingMessage.build_embeddable_contact_mention( - user_huid, - ) - - assert embeddable_mention.startswith(f" None: - chat_id = uuid.uuid4() - - embeddable_mention = SendingMessage.build_embeddable_chat_mention(chat_id) - - assert embeddable_mention.startswith(f" None: - channel_id = uuid.uuid4() - - embeddable_mention = SendingMessage.build_embeddable_channel_mention( - channel_id, - ) - - assert embeddable_mention.startswith( - f" None: - _, found_mentions = sending_message._find_and_replace_embed_mentions( - embed_mention_with_name, - ) - - assert len(found_mentions) == 1 - - def test_replace_embed_mention_without_name( - self, - sending_message: SendingMessage, - embed_mention_without_name: str, - ) -> None: - _, found_mentions = sending_message._find_and_replace_embed_mentions( - embed_mention_without_name, - ) - - assert len(found_mentions) == 1 - - @pytest.mark.parametrize( - ("embed_mention_with_name", "embed_mention_without_name"), - [ - (embed_mention_with_name, embed_mention_without_name), - (embed_mention_without_name, embed_mention_with_name), - ], - indirect=True, - ) - def test_replace_group_embed_mentions( - self, - sending_message: SendingMessage, - embed_mention_with_name: str, - embed_mention_without_name: str, - ) -> None: - embed_mentions = ", ".join( - [embed_mention_with_name, embed_mention_without_name], - ) - _, found_mentions = sending_message._find_and_replace_embed_mentions( - embed_mentions, - ) - - assert len(found_mentions) == 2 - - def test_replace_embed_mention_with_wrong_type( - self, - sending_message: SendingMessage, - embed_mention_with_wrong_type: str, - ) -> None: - with pytest.raises(ValueError): - _, found_mentions = sending_message._find_and_replace_embed_mentions( - embed_mention_with_wrong_type, - ) - - class TestAddingRecipients: - def test_adding_recipients_separately( - self, - sending_message: SendingMessage, - ) -> None: - users = [uuid.uuid4(), uuid.uuid4()] - sending_message.payload.options.recipients = "all" - - sending_message.add_recipient(users[0]) - assert sending_message.options.recipients == [users[0]] - - sending_message.add_recipient(users[1]) - assert sending_message.options.recipients == users - - def test_adding_multiple_recipients( - self, - sending_message: SendingMessage, - ) -> None: - users = [uuid.uuid4(), uuid.uuid4()] - sending_message.payload.options.recipients = "all" - - sending_message.add_recipients(users[:1]) - assert sending_message.options.recipients == users[:1] - - sending_message.add_recipients(users[1:]) - assert sending_message.options.recipients == users - - class TestMarkupAdding: - def test_adding_bubbles(self, sending_message: SendingMessage) -> None: - bubble = BubbleElement(command="/test") - sending_message.markup = MessageMarkup() - sending_message.add_bubble("/test") - sending_message.add_bubble("/test", new_row=False) - sending_message.add_bubble("/test") - sending_message.add_bubble("/test") - sending_message.add_bubble("/test", new_row=False) - assert sending_message.markup == MessageMarkup( - bubbles=[[bubble, bubble], [bubble], [bubble, bubble]], - ) - - def test_adding_keyboard(self, sending_message: SendingMessage) -> None: - keyboard_button = KeyboardElement(command="/test") - sending_message.markup = MessageMarkup() - sending_message.add_keyboard_button("/test") - sending_message.add_keyboard_button("/test", new_row=False) - sending_message.add_keyboard_button("/test") - sending_message.add_keyboard_button("/test") - sending_message.add_keyboard_button("/test", new_row=False) - assert sending_message.markup == MessageMarkup( - keyboard=[ - [keyboard_button, keyboard_button], - [keyboard_button], - [keyboard_button, keyboard_button], - ], - ) - - def test_setting_notification_show(self, sending_message: SendingMessage) -> None: - sending_message.show_notification(True) - assert sending_message.options.notifications.send - - def test_setting_dnd(self, sending_message: SendingMessage) -> None: - sending_message.force_notification(True) - assert sending_message.options.notifications.force_dnd - - -def test_credentials_or_parameters_required_for_message_creation(): - with pytest.raises(AssertionError): - SendingMessage() - - -class TestIsForward: - def test_is_forward_message(self, message, bot) -> None: - builder = MessageBuilder() - builder.forward(message=message) - new_message = Message.from_dict(message=builder.message.dict(), bot=bot) - assert new_message.is_forward - - def test_is_forward_message_error(self, message, bot) -> None: - assert not message.is_forward diff --git a/tests/test_models/test_messages/test_receiving.py b/tests/test_models/test_messages/test_receiving.py deleted file mode 100644 index 30f1103a..00000000 --- a/tests/test_models/test_messages/test_receiving.py +++ /dev/null @@ -1,136 +0,0 @@ -import uuid -from datetime import datetime as dt, timezone as tz -from typing import List - -import pytest - -from botx import ChatTypes, CommandTypes, EntityTypes -from botx.models.messages.incoming_message import Command, IncomingMessage, Sender - - -@pytest.mark.parametrize( - ("body", "command", "arguments", "single_argument"), - [ - ("/command", "/command", (), ""), - ("/command ", "/command", (), ""), - ("/command arg", "/command", ("arg",), "arg"), - ("/command arg ", "/command", ("arg",), "arg"), - ("/command \t\t arg ", "/command", ("arg",), "arg"), - ("/command arg arg", "/command", ("arg", "arg"), "arg arg"), - ("/command arg arg ", "/command", ("arg", "arg"), "arg arg"), - ], -) -def test_command_splits_right( - body: str, - command: str, - arguments: List[str], - single_argument: str, -) -> None: - command = Command(body=body, command_type=CommandTypes.user) - assert command.body == body - assert command.command == command.command - assert command.arguments == arguments - assert command.single_argument == command.single_argument - - -def test_command_data_as_dict() -> None: - command = Command( - body="/test", - command_type=CommandTypes.user, - data={"some": "data"}, - ) - assert command.data_dict == command.data == {"some": "data"} - - -def test_user_email_when_credentials_passed() -> None: - sender = Sender( - user_huid=uuid.uuid4(), - group_chat_id=uuid.uuid4(), - chat_type=ChatTypes.chat, - ad_login="user", - ad_domain="example.com", - username="test user", - is_admin=False, - is_creator=True, - host="cts.example.com", - ) - assert sender.upn == "user@example.com" - - -def test_user_email_when_credentials_missed() -> None: - assert ( - Sender( - group_chat_id=uuid.uuid4(), - chat_type=ChatTypes.chat, - is_admin=False, - is_creator=True, - host="cts.example.com", - ).upn - is None - ) - - -def test_skip_validation_for_file() -> None: - file_data = {"file_name": "zen.py", "data": "data:text/plain;base64,"} - - IncomingMessage.parse_obj( - { - "sync_id": "a465f0f3-1354-491c-8f11-f400164295cb", - "command": {"body": "/cmd", "command_type": "user", "data": {}}, - "file": file_data, - "from": { - "user_huid": None, - "group_chat_id": "8dada2c8-67a6-4434-9dec-570d244e78ee", - "ad_login": None, - "ad_domain": None, - "username": None, - "chat_type": "group_chat", - "host": "cts.ccteam.ru", - "is_admin": False, - "is_creator": False, - }, - "bot_id": "dcfa5a7c-7cc4-4c89-b6c0-80325604f9f4", - }, - ) - - -def test_parse_message_forward() -> None: - inserted_at = dt(2020, 7, 10, 10, 12, 58, 420000, tzinfo=tz.utc) - - IncomingMessage.parse_obj( - { - "bot_id": "f6615a30-9d3d-5770-b453-749ea562a974", - "command": { - "body": "Message body", - "command_type": CommandTypes.user, - "data": {}, - "metadata": {}, - }, - "entities": [ - { - "data": { - "forward_type": ChatTypes.chat, - "group_chat_id": "b51df4c1-3834-0949-1066-614ec424d28a", - "sender_huid": "4471289e-5b52-5c1b-8eab-a22c548fef9b", - "source_chat_name": "MessageAuthor Name", - "source_inserted_at": inserted_at, - "source_sync_id": "80d2c3a9-0031-50a8-aeed-32bb5d285758", - }, - "type": EntityTypes.forward, - }, - ], - "file": None, - "sync_id": "eeb8eeca-3f31-5037-8b41-84de63909a31", - "user": { - "ad_domain": "ccsteam.ru", - "ad_login": "message.forwarder", - "chat_type": ChatTypes.chat, - "group_chat_id": "070d866f-fe5b-0222-2a9e-b7fc35c99465", - "host": "cts.ccsteam.ru", - "is_admin": True, - "is_creator": True, - "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", - "username": "MessageForwarder Name", - }, - }, - ) diff --git a/tests/test_models/test_messages/test_sending.py b/tests/test_models/test_messages/test_sending.py deleted file mode 100644 index 9ef90307..00000000 --- a/tests/test_models/test_messages/test_sending.py +++ /dev/null @@ -1,35 +0,0 @@ -from botx import BubbleElement, KeyboardElement, MessageMarkup, UpdatePayload - - -def test_message_markup_will_add_row_if_there_is_no_existed_and_not_new_row() -> None: - markup1 = MessageMarkup() - markup1.add_bubble("/command", new_row=False) - - markup2 = MessageMarkup() - markup2.add_bubble("/command") - - assert markup1 == markup2 - - -def test_update_markup_will_can_be_set_from_markup() -> None: - markup = MessageMarkup() - markup.add_bubble("/command") - markup.add_keyboard_button("/command") - - update = UpdatePayload() - update.set_markup(markup) - - assert update.markup == markup - assert update.bubbles == markup.bubbles - assert update.keyboard == markup.keyboard - - -def test_adding_markup_by_elements() -> None: - bubble = BubbleElement(command="/command") - keyboard = KeyboardElement(command="/command") - - markup = MessageMarkup() - markup.add_bubble_element(bubble) - markup.add_keyboard_button_element(keyboard) - - assert markup == MessageMarkup(bubbles=[[bubble]], keyboard=[[keyboard]]) diff --git a/tests/test_models/test_smartapps.py b/tests/test_models/test_smartapps.py deleted file mode 100644 index ff210fcc..00000000 --- a/tests/test_models/test_smartapps.py +++ /dev/null @@ -1,141 +0,0 @@ -from io import BytesIO -from typing import Any, Dict -from uuid import UUID - -from botx import File, Message -from botx.models.smartapps import SendingSmartAppEvent, SendingSmartAppNotification - -pytest_plugins = ("tests.test_clients.fixtures", "tests.fixtures.smartapps") - - -def test_sending_smartapp_event( - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -): - sending_smartapp = SendingSmartAppEvent( - ref=ref, - smartapp_id=smartapp_id, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - data=smartapp_data, - ) - - assert sending_smartapp.ref == ref - assert sending_smartapp.smartapp_id == smartapp_id - assert sending_smartapp.smartapp_api_version == smartapp_api_version - assert sending_smartapp.group_chat_id == group_chat_id - assert sending_smartapp.data == smartapp_data - - -def test_sending_smartapp_notification( - ref: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_counter: int, -): - sending_smartapp = SendingSmartAppNotification( - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - smartapp_counter=smartapp_counter, - ) - - assert sending_smartapp.smartapp_api_version == smartapp_api_version - assert sending_smartapp.group_chat_id == group_chat_id - assert sending_smartapp.smartapp_counter == smartapp_counter - - -def test_sending_smartapp_notification_from_message( - message: Message, - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_counter: int, -): - message.incoming_message.command.data_dict[ - "smartapp_api_version" - ] = smartapp_api_version - message.incoming_message.command.data_dict["opts"] = {} - message.group_chat_id = group_chat_id - - sending_smartapp = SendingSmartAppNotification.from_message( - smartapp_counter=smartapp_counter, - message=message, - ) - - assert sending_smartapp.smartapp_api_version == smartapp_api_version - assert sending_smartapp.group_chat_id == group_chat_id - assert sending_smartapp.smartapp_counter == smartapp_counter - - -def test_sending_smartapp_event_from_message( - message: Message, - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -): - message.incoming_message.command.data_dict[ - "smartapp_api_version" - ] = smartapp_api_version - message.incoming_message.command.data_dict["opts"] = {} - message.incoming_message.command.data_dict["smartapp_id"] = smartapp_id - message.incoming_message.command.data_dict["ref"] = ref - - message.group_chat_id = group_chat_id - - sending_smartapp = SendingSmartAppEvent.from_message( - data=smartapp_data, - message=message, - ) - - assert sending_smartapp.smartapp_api_version == smartapp_api_version - assert sending_smartapp.group_chat_id == group_chat_id - assert sending_smartapp.data == smartapp_data - - -def test_sending_smartapp_event_add_botx_file( - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -): - sending_smartapp = SendingSmartAppEvent( - ref=ref, - smartapp_id=smartapp_id, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - data=smartapp_data, - ) - - file = File.from_string(b"data", filename="file.txt") - sending_smartapp.add_file(file) - - assert sending_smartapp.files == [file] - - -def test_sending_smartapp_event_add_file( - ref: UUID, - smartapp_id: UUID, - smartapp_api_version: int, - group_chat_id: UUID, - smartapp_data: Dict[str, Any], -): - sending_smartapp = SendingSmartAppEvent( - ref=ref, - smartapp_id=smartapp_id, - smartapp_api_version=smartapp_api_version, - group_chat_id=group_chat_id, - data=smartapp_data, - ) - - file_data = b"data" - file = File.from_string(file_data, filename="file.txt") - sending_smartapp.add_file(BytesIO(file_data), file.file_name) - - assert sending_smartapp.files == [file] diff --git a/tests/test_models/test_status.py b/tests/test_models/test_status.py deleted file mode 100644 index 53df32c0..00000000 --- a/tests/test_models/test_status.py +++ /dev/null @@ -1,27 +0,0 @@ -import uuid - -from botx import ChatTypes -from botx.models.status import StatusRecipient - - -def test_status_recipient(): - bot_id = uuid.uuid4() - user_huid = uuid.uuid4() - ad_login = "login" - ad_domain = "domain" - is_admin = True - chat_type = ChatTypes.chat - recipient = StatusRecipient( - bot_id=bot_id, - user_huid=user_huid, - ad_login=ad_login, - ad_domain=ad_domain, - is_admin=is_admin, - chat_type=chat_type, - ) - assert recipient.bot_id == bot_id - assert recipient.user_huid == user_huid - assert recipient.ad_login == ad_login - assert recipient.ad_domain == ad_domain - assert recipient.is_admin == is_admin - assert recipient.chat_type == chat_type diff --git a/tests/test_state.py b/tests/test_state.py new file mode 100644 index 00000000..35ed2b3d --- /dev/null +++ b/tests/test_state.py @@ -0,0 +1,93 @@ +from typing import Callable, Optional + +import pytest + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + IncomingMessageHandlerFunc, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__bot_state__save_changes_between_middleware_and_handler( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + user_command = incoming_message_factory(body="/command") + + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + bot.state.api_token = "token" + + await call_next(message, bot) + + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + middlewares=[middleware], + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert built_bot.state.api_token == "token" + + +async def test__message_state__save_changes_between_middleware_and_handler( + incoming_message_factory: Callable[..., IncomingMessage], + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + incoming_message: Optional[IncomingMessage] = None + user_command = incoming_message_factory(body="/command") + + async def middleware( + message: IncomingMessage, + bot: Bot, + call_next: IncomingMessageHandlerFunc, + ) -> None: + message.state.username = "ivanov_ivan_1990" + + await call_next(message, bot) + + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + nonlocal incoming_message + incoming_message = message + + built_bot = Bot( + collectors=[collector], + bot_accounts=[bot_account], + middlewares=[middleware], + ) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(user_command) + + # - Assert - + assert incoming_message is not None + assert incoming_message.state.username == "ivanov_ivan_1990" diff --git a/tests/test_status.py b/tests/test_status.py new file mode 100644 index 00000000..760d8aea --- /dev/null +++ b/tests/test_status.py @@ -0,0 +1,319 @@ +from unittest.mock import Mock +from uuid import UUID, uuid4 + +import pytest + +from botx import ( + Bot, + BotAccountWithSecret, + BotMenu, + ChatTypes, + HandlerCollector, + IncomingMessage, + StatusRecipient, + UnknownBotAccountError, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +@pytest.fixture +def status_recipient(bot_id: UUID) -> StatusRecipient: + return StatusRecipient( + bot_id=bot_id, + huid=uuid4(), + ad_login=None, + ad_domain=None, + is_admin=True, + chat_type=ChatTypes.PERSONAL_CHAT, + ) + + +async def test__get_status__hidden_command_not_in_menu( + bot_account: BotAccountWithSecret, + status_recipient: StatusRecipient, + incorrect_handler_trigger: Mock, +) -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.command("/_command", visible=False) + async def handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.get_status(status_recipient) + + # - Assert - + assert status == BotMenu({}) + + incorrect_handler_trigger.assert_not_called() + + +async def test__get_status__visible_command_in_menu( + bot_account: BotAccountWithSecret, + status_recipient: StatusRecipient, + incorrect_handler_trigger: Mock, +) -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.command("/command", description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.get_status(status_recipient) + + # - Assert - + assert status == BotMenu({"/command": "My command"}) + + incorrect_handler_trigger.assert_not_called() + + +async def test__get_status__command_not_in_menu_if_visible_func_return_false( + bot_account: BotAccountWithSecret, + status_recipient: StatusRecipient, + incorrect_handler_trigger: Mock, +) -> None: + # - Arrange - + collector = HandlerCollector() + + async def visible_func(status_recipient: StatusRecipient, bot: Bot) -> bool: + return False + + @collector.command("/command", visible=visible_func, description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.get_status(status_recipient) + + # - Assert - + assert status == BotMenu({}) + + incorrect_handler_trigger.assert_not_called() + + +async def test__get_status__command_in_menu_if_visible_func_return_true( + bot_account: BotAccountWithSecret, + status_recipient: StatusRecipient, + incorrect_handler_trigger: Mock, +) -> None: + # - Arrange - + collector = HandlerCollector() + + async def visible_func(status_recipient: StatusRecipient, bot: Bot) -> bool: + return True + + @collector.command("/command", visible=visible_func, description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + incorrect_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.get_status(status_recipient) + + # - Assert - + assert status == BotMenu({"/command": "My command"}) + + incorrect_handler_trigger.assert_not_called() + + +async def test__raw_get_status__invalid_query() -> None: + # - Arrange - + query = {"user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11"} + + collector = HandlerCollector() + + @collector.command("/_command", visible=False) + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + built_bot = Bot(collectors=[collector], bot_accounts=[]) + + # - Act - + with pytest.raises(ValueError) as exc: + async with lifespan_wrapper(built_bot) as bot: + await bot.raw_get_status(query) + + # - Assert - + assert "validation error" in str(exc.value) + + +async def test__raw_get_status__unknown_bot_account_error_raised() -> None: + # - Arrange - + query = { + "bot_id": "123e4567-e89b-12d3-a456-426655440000", + "chat_type": "chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + with pytest.raises(UnknownBotAccountError) as exc: + await bot.raw_get_status(query) + + # - Assert - + assert "123e4567-e89b-12d3-a456-426655440000" in str(exc.value) + + +async def test__raw_get_status__minimally_filled_succeed( + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + query = { + "bot_id": str(bot_id), + "chat_type": "chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.raw_get_status(query) + + # - Assert - + assert status + + +async def test__raw_get_status__minimum_filled_succeed( + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + query = { + "ad_domain": "", + "ad_login": "", + "is_admin": "", + "bot_id": str(bot_id), + "chat_type": "group_chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.raw_get_status(query) + + # - Assert - + assert status + + +async def test__raw_get_status__maximum_filled_succeed( + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + query = { + "ad_domain": "domain", + "ad_login": "login", + "bot_id": str(bot_id), + "chat_type": "chat", + "is_admin": "true", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + built_bot = Bot(collectors=[HandlerCollector()], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.raw_get_status(query) + + # - Assert - + assert status + + +async def test__raw_get_status__hidden_command_not_in_status( + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + query = { + "bot_id": str(bot_id), + "chat_type": "chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + collector = HandlerCollector() + + @collector.command("/_command", visible=False) + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.raw_get_status(query) + + # - Assert - + assert status == { + "result": { + "commands": [], + "enabled": True, + "status_message": "Bot is working", + }, + "status": "ok", + } + + +async def test__raw_get_status__visible_command_in_status( + bot_account: BotAccountWithSecret, + bot_id: UUID, +) -> None: + # - Arrange - + query = { + "bot_id": str(bot_id), + "chat_type": "chat", + "user_huid": "f16cdc5f-6366-5552-9ecd-c36290ab3d11", + } + + collector = HandlerCollector() + + @collector.command("/command", visible=True, description="My command") + async def handler(message: IncomingMessage, bot: Bot) -> None: + pass + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + status = await bot.raw_get_status(query) + + # - Assert - + assert status == { + "result": { + "commands": [ + { + "body": "/command", + "description": "My command", + "name": "/command", + }, + ], + "enabled": True, + "status_message": "Bot is working", + }, + "status": "ok", + } diff --git a/tests/test_stickers.py b/tests/test_stickers.py new file mode 100644 index 00000000..32de09e2 --- /dev/null +++ b/tests/test_stickers.py @@ -0,0 +1,77 @@ +from http import HTTPStatus +from typing import Any, Callable, Dict +from uuid import UUID + +import httpx +import pytest +from aiofiles.tempfile import NamedTemporaryFile +from respx.router import MockRouter + +from botx import ( + Bot, + BotAccountWithSecret, + HandlerCollector, + IncomingMessage, + Sticker, + lifespan_wrapper, +) + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + +PNG_IMAGE = ( + b"\x89PNG\r\n\x1a\n" + b"\x00\x00\x00\rIHDR\x00\x00\x00\x01\x00\x00\x00\x01\x01\x03\x00" + b"\x00\x00%\xdbV\xca\x00\x00\x00\x03PLTE\x00\x00\x00\xa7z=\xda\x00" + b"\x00\x00\x01tRNS\x00@\xe6\xd8f\x00\x00\x00\nIDAT\x08\xd7c`\x00" + b"\x00\x00\x02\x00\x01\xe2!\xbc3\x00\x00\x00\x00IEND\xaeB`\x82" +) + + +async def test__sticker__download( + respx_mock: MockRouter, + host: str, + bot_account: BotAccountWithSecret, + async_buffer: NamedTemporaryFile, + api_incoming_message_factory: Callable[..., Dict[str, Any]], +) -> None: + # - Arrange - + image_link = ( + f"https://{host}/uploads/sticker_pack/" + "4ff8113b-8460-5977-86b2-c1798eb4fbce/" + "14a762edf2e04c579de98098e22b01da.png" + ) + + endpoint = respx_mock.get(image_link).mock( + return_value=httpx.Response( + HTTPStatus.OK, + content=PNG_IMAGE, + ), + ) + + sticker = Sticker( + id=UUID("4ff8113b-8460-5977-86b2-c1798eb4fbce"), + emoji="🤔", + image_link=image_link, + ) + + payload = api_incoming_message_factory() + + collector = HandlerCollector() + + @collector.default_message_handler + async def default_handler(message: IncomingMessage, bot: Bot) -> None: + await sticker.download(async_buffer) + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_raw_bot_command(payload) + + # - Assert - + assert await async_buffer.read() == PNG_IMAGE + assert endpoint.called diff --git a/tests/test_system_events_routing.py b/tests/test_system_events_routing.py new file mode 100644 index 00000000..df41ce2a --- /dev/null +++ b/tests/test_system_events_routing.py @@ -0,0 +1,138 @@ +from unittest.mock import Mock +from uuid import UUID, uuid4 + +import pytest + +from botx import ( + Bot, + BotAccount, + BotAccountWithSecret, + Chat, + ChatCreatedEvent, + ChatCreatedMember, + ChatTypes, + HandlerCollector, + UserKinds, + lifespan_wrapper, +) + + +@pytest.fixture +def chat_created( + bot_id: UUID, +) -> ChatCreatedEvent: + return ChatCreatedEvent( + bot=BotAccount( + id=bot_id, + host="cts.example.com", + ), + sync_id=uuid4(), + chat_name="Test", + chat=Chat( + id=uuid4(), + type=ChatTypes.PERSONAL_CHAT, + ), + creator_id=uuid4(), + members=[ + ChatCreatedMember( + is_admin=False, + huid=uuid4(), + username="Ivanov Ivan Ivanovich", + kind=UserKinds.CTS_USER, + ), + ], + raw_command=None, + ) + + +pytestmark = [ + pytest.mark.asyncio, + pytest.mark.mock_authorization, + pytest.mark.usefixtures("respx_mock"), +] + + +async def test__system_event_handler__called( + chat_created: ChatCreatedEvent, + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + collector = HandlerCollector() + + @collector.chat_created + async def chat_created_handler(event: ChatCreatedEvent, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(chat_created) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +async def test__system_event_handler__no_handler_for_system_event( + chat_created: ChatCreatedEvent, + bot_account: BotAccountWithSecret, + loguru_caplog: pytest.LogCaptureFixture, +) -> None: + # - Arrange - + collector = HandlerCollector() + + built_bot = Bot(collectors=[collector], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(chat_created) + + # - Assert - + assert "Handler for `ChatCreatedEvent` not found" in loguru_caplog.text + + +async def test__system_event_handler__handler_in_first_collector( + chat_created: ChatCreatedEvent, + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + collector_1 = HandlerCollector() + collector_2 = HandlerCollector() + + @collector_1.chat_created + async def chat_created_handler(event: ChatCreatedEvent, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector_1, collector_2], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(chat_created) + + # - Assert - + correct_handler_trigger.assert_called_once() + + +async def test__system_event_handler__handler_in_second_collector( + chat_created: ChatCreatedEvent, + correct_handler_trigger: Mock, + bot_account: BotAccountWithSecret, +) -> None: + # - Arrange - + collector_1 = HandlerCollector() + collector_2 = HandlerCollector() + + @collector_2.chat_created + async def chat_created_handler(event: ChatCreatedEvent, bot: Bot) -> None: + correct_handler_trigger() + + built_bot = Bot(collectors=[collector_1, collector_2], bot_accounts=[bot_account]) + + # - Act - + async with lifespan_wrapper(built_bot) as bot: + bot.async_execute_bot_command(chat_created) + + # - Assert - + correct_handler_trigger.assert_called_once() diff --git a/tests/test_testing/__init__.py b/tests/test_testing/__init__.py deleted file mode 100644 index e69de29b..00000000 diff --git a/tests/test_testing/test_builder.py b/tests/test_testing/test_builder.py deleted file mode 100644 index f5383ea8..00000000 --- a/tests/test_testing/test_builder.py +++ /dev/null @@ -1,199 +0,0 @@ -import uuid -from datetime import datetime -from io import StringIO - -import pytest -from pydantic import ValidationError - -from botx import ( - ChatTypes, - Entity, - EntityTypes, - File, - Forward, - Mention, - MentionTypes, - MessageBuilder, - UserMention, -) -from botx.models.entities import Reply - - -def test_setting_new_file_from_file(): - file = File.from_string("some data", "name.txt") - builder = MessageBuilder() - builder.file = file - - assert builder.file == file - - -def test_setting_new_file_from_io(): - file = File.from_string("some data", "name.txt") - builder = MessageBuilder() - builder.file = file.file - - assert builder.file == file - - -def test_settings_new_user_for_message(incoming_message): - builder = MessageBuilder() - builder.user = incoming_message.user - - assert builder.user == incoming_message.user - - -def test_file_transfer_event(): - builder = MessageBuilder() - builder.file = File.from_string("some data", "name.txt") - - builder.body = "file_transfer" - builder.system_command = True - - -def test_setting_not_processable_file_for_incoming_message(): - file = StringIO("import this") - file.name = "zen.py" - - builder = MessageBuilder() - builder.file = file - - message = builder.message - - assert message.file.file_name == "zen.py" - - -def test_mention_user_in_message(): - user_huid = uuid.uuid4() - builder = MessageBuilder() - builder.mention_user(user_huid) - - assert builder.message.entities.__root__[0].data.mention_type == MentionTypes.user - assert builder.message.entities.__root__[0].data.mention_data.user_huid == user_huid - - -def test_mention_contact_in_message(): - user_huid = uuid.uuid4() - builder = MessageBuilder() - builder.mention_contact(user_huid) - - assert ( - builder.message.entities.__root__[0].data.mention_type == MentionTypes.contact - ) - assert builder.message.entities.__root__[0].data.mention_data.user_huid == user_huid - - -def test_mention_chat_in_message(): - chat_id = uuid.uuid4() - builder = MessageBuilder() - builder.mention_chat(chat_id) - - assert builder.message.entities.__root__[0].data.mention_type == MentionTypes.chat - assert ( - builder.message.entities.__root__[0].data.mention_data.group_chat_id == chat_id - ) - - -def test_setting_raw_entities(): - builder = MessageBuilder() - builder.entities = [ - Entity( - type=EntityTypes.mention, - data=Mention(mention_data=UserMention(user_huid=uuid.uuid4())), - ), - ] - - assert builder.message.entities.__root__[0].data.mention_type == MentionTypes.user - - -@pytest.mark.parametrize( - "include_param", - ["user_huid", "ad_login", "ad_domain", "username"], -) -def test_error_when_chat_validation_not_passed(include_param): - user_params = {"user_huid", "ad_login", "ad_domain", "username"} - builder = MessageBuilder() - - builder.body = "system:chat_created" - builder.user = builder.user.copy( - update={param: None for param in user_params - {include_param}}, - ) - builder.command_data = { - "group_chat_id": uuid.uuid4(), - "chat_type": "group_chat", - "name": "", - "creator": uuid.uuid4(), - "members": [], - } - with pytest.raises(ValidationError): - builder.system_command = True - - -def test_error_when_file_validation_not_passed(): - builder = MessageBuilder() - builder.body = "file_transfer" - with pytest.raises(ValidationError): - builder.system_command = True - - -class TestBuildForward: - def test_building_forward_via_models(self): - builder = MessageBuilder() - builder.forward( - forward=Forward( - group_chat_id=uuid.uuid4(), - sender_huid=uuid.uuid4(), - forward_type=ChatTypes.group_chat, # ignore: type - source_inserted_at=datetime.now(), - source_sync_id=uuid.uuid4(), - ), - ) - - def test_building_forward_via_message(self, message): - builder = MessageBuilder() - builder.forward(message=message) - - def test_building_forward_arguments_error(self, message): - builder = MessageBuilder() - with pytest.raises(ValueError): - builder.forward( - message=message, - forward=Forward( - group_chat_id=uuid.uuid4(), - sender_huid=uuid.uuid4(), - forward_type=ChatTypes.botx, # ignore: type - source_inserted_at=datetime.now(), - source_sync_id=uuid.uuid4(), - ), - ) - - -class TestBuildReply: - def test_building_forward_via_models(self): - builder = MessageBuilder() - builder.reply( - reply=Reply( - body="foo", - reply_type=ChatTypes.group_chat, - sender=uuid.uuid4(), - source_chat_name="bar", - source_sync_id=uuid.uuid4(), - ), - ) - - def test_building_forward_via_message(self, message): - builder = MessageBuilder() - builder.reply(message=message) - - def test_building_forward_arguments_error(self, message): - builder = MessageBuilder() - with pytest.raises(ValueError): - builder.reply( - message=message, - reply=Reply( - body="foo", - reply_type=ChatTypes.group_chat, - sender=uuid.uuid4(), - source_chat_name="bar", - source_sync_id=uuid.uuid4(), - ), - ) diff --git a/tests/test_testing/test_client.py b/tests/test_testing/test_client.py deleted file mode 100644 index 9eafbf8b..00000000 --- a/tests/test_testing/test_client.py +++ /dev/null @@ -1,20 +0,0 @@ -import threading - -import pytest - -from botx import TestClient - -pytestmark = pytest.mark.asyncio - - -async def test_disabling_sync_send_for_client(bot, incoming_message, build_handler): - bot.default(build_handler(threading.Event())) - - with TestClient(bot) as client: - await client.send_command(incoming_message, False) - - assert bot.tasks - - await bot.shutdown() - - assert not bot.tasks