Skip to main content

Modern python http client

Project description

Retejo

Функции

  • Валидация с помощью adaptix(и не только): Используйте библиотеку adaptix для парсинга ответов. Вы можете с легкостью заменить adaptix на pydantic.
  • Интуитивно понятный интерфейс: Полностью типизированный клиент упрощает процесс разработки, устроняя множество ошибок ещё до запуска кода.
  • Интеграции: Retejo поддерживает интеграцию со такими клиентами как: requests и aiohttp.

Установка

Вы можете установить retejo с помощью pip:

pip install retejo[requests]
pip install retejo[aiohttp]

или uv:

uv pip install retejo[requests]
uv pip install retejo[aiohttp]

Написание кода.

Для начала нужно объявить модель, например, dataclass.

@dataclass
class Post:
    id: int
    title: str
    body: str
    user_id: int

@dataclass
class PostId:
    id: int

Далее объявить метод.

В retejo каждый метод наследуется от базового класса Method. При наследовании в Method передается модель, которая описывает ответ метода.

Так же, нужно указать часть пусти к ендпоинту и тип метода.

class GetPost(Method[Post]):
    __url__ = "posts/{id}"
    __method__ = "get"

    id: UrlVar[int]

class CreatePost(Method[PostId]):
    __url__ = "posts"
    __method__ = "post"

    user_id: Body[int]
    title: Body[str]
    body: Body[str]

Теперь клиент. В super передаем базовую ссылку.

Так же переопределяем логику парсинга ответа. За большей информацией по adaptix обратитесь к его документации.

class Client(RequestsClient):
    def __init__(self) -> None:
        super().__init__("https://jsonplaceholder.typicode.com/")

    @override
    def init_response_factory(self) -> Factory:
        return Retort(
            recipe=[
                # поля в ответе вида camelCase
                # будут конвертированы в lower_case.
                name_mapping(
                    name_style=NameStyle.CAMEL,
                ),
            ]
        )

Далее нужно привязать методы к клиенту. Для этого используется функция bind_method.

class Client(RequestsClient):
    # ...
    get_post = bind_method(GetPost)
    create_post = bind_method(CreatePost)

Вы наверное подумаете, что из за этого не будет работать типизация. Наоборот, поведение будет как у метода, которая принимает аргументы для инициализации метода и возвращает ответ.

Например строка.

get_post = bind_method(GetPost)

Эквивалентно следующему.

def get_post(self, id: int) -> Post:
    return self.send_method(
        GetPost(
            id=id,
        ),
    )

Типизация работает на все 100%.

И, конечно же, использование клиента.

client = Client()
created_post = client.create_post(
    user_id=1,
    title="Title",
    body="Body"
)
got_post = client.get_post(created_post.id)
client.close()

Весь код.

@dataclass
class Post:
    id: int
    title: str
    body: str
    user_id: int

@dataclass
class PostId:
    id: int

class GetPost(Method[Post]):
    __url__ = "posts/{id}"
    __method__ = "get"

    id: UrlVar[int]

class CreatePost(Method[PostId]):
    __url__ = "posts"
    __method__ = "post"

    user_id: Body[int]
    title: Body[str]
    body: Body[str]


class Client(RequestsClient):
    def __init__(self) -> None:
        super().__init__(base_url="https://jsonplaceholder.typicode.com/")

    @override
    def init_response_factory(self) -> Factory:
        return Retort(
            recipe=[
                name_mapping(name_style=NameStyle.CAMEL),
            ]
        )

    get_post = bind_method(GetPost)
    create_post = bind_method(CreatePost)


client = Client()
created_post = client.create_post(
    user_id=1,
    title="Title",
    body="Body"
)
got_post = client.get_post(created_post.id)
client.close()

Асинхронность.

Для того, чтобы использовать асинхронный подход, нужно:

  1. Установить retejo с aiohttp.

    pip install retejo[aiohttp]
    
  2. Наследоваться от AiohttpClient.

  3. Вызывать все методы клиента асинхронно.

    class Client(AiohttpClient):
        def __init__(self) -> None:
            super().__init__(base_url="https://jsonplaceholder.typicode.com/")
    
    @override
    def init_response_factory(self) -> Factory:
        return Retort(
            recipe=[
                name_mapping(name_style=NameStyle.CAMEL),
            ]
        )
    
        get_post = bind_method(GetPost)
        create_post = bind_method(CreatePost)
    
    
    client = Client()
    created_post = await client.create_post(
        user_id=1,
        title="Title",
        body="Body"
    )
    got_post = await client.get_post(created_post.id)
    client.close()
    

Маркеры

Retejo предоставляет несколько базовых маркеров для ваших методов:

  • Body - параметр должен передано в тело запроса.

  • File - параметр является файлом.

  • Header - параметр должен передано в заголовки запроса.

  • QueryParam - параметр должен передано в параметры запроса.

  • UrlVar - параметр используется для форматирования ссылки.

  • Omittable - этот флаг комбинируется с выше перечисленными.

    Если значение параметра является объект Omitted, то это поле не попадет в запрос.

Пример использования.

class AddModel(Method[Any]):
    __url__ = "user/{user_id}/models"
    __method__ = "post"

    user_id: UrlVar[int]
    model: Body[str]
    access_token: Header[str]
    number: Body[Omittable[str]] = Omitted()


class Client(RequestsClient):
    def __init__(self) -> None:
        super().__init__("")

    @override
    def init_markers_factories(self) -> MarkersFactorties:
        factories = super().init_markers_factories()
        factories[HeaderMarker] =  Retort(
            recipe=[
                dumper(
                    P[AddModel].access_token,
                    lambda x: f"Bearer {x}",
                ),
                name_mapping(
                    AddModel,
                    map={
                        "access_token": "Authorization",
                    },
                ),
            ],
        )
        return factories

    add_model = bind_method(AddModel)

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

retejo-0.0.2.tar.gz (28.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

retejo-0.0.2-py3-none-any.whl (35.6 kB view details)

Uploaded Python 3

File details

Details for the file retejo-0.0.2.tar.gz.

File metadata

  • Download URL: retejo-0.0.2.tar.gz
  • Upload date:
  • Size: 28.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.7.3

File hashes

Hashes for retejo-0.0.2.tar.gz
Algorithm Hash digest
SHA256 2e80e019a02177a4486106947e5ca8cb5e6c21e0c8ed5190e3e8daaa2fc72a96
MD5 4142f3afa0a130fa9ce29d66d1f8773d
BLAKE2b-256 0f6e0566c9f1b4838c6f3d6003a5cb6f4c86c99ba1eab8670b0d2f77f4d572ee

See more details on using hashes here.

File details

Details for the file retejo-0.0.2-py3-none-any.whl.

File metadata

  • Download URL: retejo-0.0.2-py3-none-any.whl
  • Upload date:
  • Size: 35.6 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.7.3

File hashes

Hashes for retejo-0.0.2-py3-none-any.whl
Algorithm Hash digest
SHA256 3218168639608ca1c269f57b17d4b29ef63e1e6fc3d978d74054a5c86644f089
MD5 b146d8b09809eda8f04798a7edaa5f9d
BLAKE2b-256 7b11d9c82da61e58bdf74904ead8c3da07ebd5cfd76199268dff12a966988c2d

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page