From b3c5b29dac0afc41c3c61a80f07354c36b451e65 Mon Sep 17 00:00:00 2001 From: Yair Siman Tov <63305203+yairsimantov20@users.noreply.github.com> Date: Thu, 21 Dec 2023 13:07:46 +0200 Subject: [PATCH] Ocean async client (#285) --- CHANGELOG.md | 13 ++++++ changelog/1.bugfix.md | 1 - changelog/PORT-5711.improvement.md | 1 - port_ocean/clients/port/retry_transport.py | 4 +- port_ocean/clients/port/utils.py | 10 ++-- port_ocean/helpers/async_client.py | 53 ++++++++++++++++++++++ port_ocean/utils.py | 10 ++-- pyproject.toml | 2 +- 8 files changed, 76 insertions(+), 18 deletions(-) delete mode 100644 changelog/1.bugfix.md delete mode 100644 changelog/PORT-5711.improvement.md create mode 100644 port_ocean/helpers/async_client.py diff --git a/CHANGELOG.md b/CHANGELOG.md index 1e08f732d5..101585f545 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,19 @@ this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.htm +## 0.4.10 (2023-12-21) + + +### Improvements + +- Wrapped the httpx async client with implementation that overrides the default transport class with custom transport to apply all default httpx features that are ignored when passing a custom transport instance. This allows the missing behevior of the http (proxy environment variable)[https://www.python-httpx.org/environment_variables/#proxies] (PORT-5676) +- Changed deprecated `poetry lock --check` in the make files to `poetry check` (PORT-5711) + +### Bug Fixes + +- Changed the way we upsert and delete bulk of entities from the catalog to be batched rather than spawning all requests at once + + ## 0.4.9 (2023-12-19) diff --git a/changelog/1.bugfix.md b/changelog/1.bugfix.md deleted file mode 100644 index a38dc09f6d..0000000000 --- a/changelog/1.bugfix.md +++ /dev/null @@ -1 +0,0 @@ -Changed the way we upsert and delete bulk of entities from the catalog to be batched rather than spawning all requests at once \ No newline at end of file diff --git a/changelog/PORT-5711.improvement.md b/changelog/PORT-5711.improvement.md deleted file mode 100644 index 9033df168b..0000000000 --- a/changelog/PORT-5711.improvement.md +++ /dev/null @@ -1 +0,0 @@ -Changed deprecated `poetry lock --check` in the make files to `poetry check` \ No newline at end of file diff --git a/port_ocean/clients/port/retry_transport.py b/port_ocean/clients/port/retry_transport.py index 62250b8655..7654585bc6 100644 --- a/port_ocean/clients/port/retry_transport.py +++ b/port_ocean/clients/port/retry_transport.py @@ -11,8 +11,8 @@ class TokenRetryTransport(RetryTransport): - def __init__(self, port_client: "PortClient", *args: Any, **kwargs: Any) -> None: - super().__init__(*args, **kwargs) + def __init__(self, port_client: "PortClient", **kwargs: Any) -> None: + super().__init__(**kwargs) self.port_client = port_client def _is_retryable_method(self, request: httpx.Request) -> bool: diff --git a/port_ocean/clients/port/utils.py b/port_ocean/clients/port/utils.py index 07b5e56402..52edbfc78d 100644 --- a/port_ocean/clients/port/utils.py +++ b/port_ocean/clients/port/utils.py @@ -5,6 +5,7 @@ from werkzeug.local import LocalStack, LocalProxy from port_ocean.clients.port.retry_transport import TokenRetryTransport +from port_ocean.helpers.async_client import OceanAsyncClient if TYPE_CHECKING: from port_ocean.clients.port.client import PortClient @@ -27,12 +28,9 @@ def _get_http_client_context(port_client: "PortClient") -> httpx.AsyncClient: client = _http_client.top if client is None: - client = httpx.AsyncClient( - transport=TokenRetryTransport( - port_client, - httpx.AsyncHTTPTransport(), - logger=logger, - ), + client = OceanAsyncClient( + TokenRetryTransport, + transport_kwargs={"port_client": port_client}, timeout=httpx.Timeout(PORT_HTTP_TIMEOUT), limits=httpx.Limits( max_connections=PORT_HTTP_MAX_CONNECTIONS_LIMIT, diff --git a/port_ocean/helpers/async_client.py b/port_ocean/helpers/async_client.py new file mode 100644 index 0000000000..c78e40e61c --- /dev/null +++ b/port_ocean/helpers/async_client.py @@ -0,0 +1,53 @@ +from typing import Any, Callable, Type + +import httpx +from loguru import logger + +from port_ocean.helpers.retry import RetryTransport + + +class OceanAsyncClient(httpx.AsyncClient): + """ + This class is a wrapper around httpx.AsyncClient that uses a custom transport class. + This is done to allow passing our custom transport class to the AsyncClient constructor while still allowing + all the default AsyncClient behavior that is changed when passing a custom transport instance. + """ + + def __init__( + self, + transport_class: Type[RetryTransport] = RetryTransport, + transport_kwargs: dict[str, Any] | None = None, + **kwargs: Any, + ): + self._transport_kwargs = transport_kwargs + self._transport_class = transport_class + super().__init__(**kwargs) + + def _init_transport( # type: ignore[override] + self, + transport: httpx.AsyncBaseTransport | None = None, + app: Callable[..., Any] | None = None, + **kwargs: Any, + ) -> httpx.AsyncBaseTransport: + if transport is not None or app is not None: + return super()._init_transport(transport=transport, app=app, **kwargs) + + return self._transport_class( + wrapped_transport=httpx.AsyncHTTPTransport( + **kwargs, + ), + logger=logger, + **(self._transport_kwargs or {}), + ) + + def _init_proxy_transport( # type: ignore[override] + self, proxy: httpx.Proxy, **kwargs: Any + ) -> httpx.AsyncBaseTransport: + return self._transport_class( + wrapped_transport=httpx.AsyncHTTPTransport( + proxy=proxy, + **kwargs, + ), + logger=logger, + **(self._transport_kwargs or {}), + ) diff --git a/port_ocean/utils.py b/port_ocean/utils.py index a5384dadc7..78ca894d1f 100644 --- a/port_ocean/utils.py +++ b/port_ocean/utils.py @@ -3,6 +3,7 @@ from asyncio import ensure_future from functools import wraps from importlib.util import module_from_spec, spec_from_file_location +from pathlib import Path from time import time from traceback import format_exception from types import ModuleType @@ -13,10 +14,10 @@ import tomli import yaml from loguru import logger -from pathlib import Path from starlette.concurrency import run_in_threadpool from werkzeug.local import LocalStack, LocalProxy +from port_ocean.helpers.async_client import OceanAsyncClient from port_ocean.helpers.retry import RetryTransport _http_client: LocalStack[httpx.AsyncClient] = LocalStack() @@ -25,12 +26,7 @@ def _get_http_client_context() -> httpx.AsyncClient: client = _http_client.top if client is None: - client = httpx.AsyncClient( - transport=RetryTransport( - httpx.AsyncHTTPTransport(), - logger=logger, - ) - ) + client = OceanAsyncClient(RetryTransport) _http_client.push(client) return client diff --git a/pyproject.toml b/pyproject.toml index afd2b737a3..6752a01065 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [tool.poetry] name = "port-ocean" -version = "0.4.9" +version = "0.4.10" description = "Port Ocean is a CLI tool for managing your Port projects." readme = "README.md" homepage = "https://app.getport.io"