From 7b42ca71cf0256b0b815340e55dd3878d1118fa1 Mon Sep 17 00:00:00 2001 From: Omer Zuarets Date: Thu, 30 May 2024 17:20:40 +0300 Subject: [PATCH] add httpx client for HttpFetchProvider and make it default --- .../fetcher/providers/http_fetch_provider.py | 39 ++++++++++++------- packages/opal-common/opal_common/http.py | 7 +++- packages/opal-common/requires.txt | 1 + 3 files changed, 31 insertions(+), 16 deletions(-) diff --git a/packages/opal-common/opal_common/fetcher/providers/http_fetch_provider.py b/packages/opal-common/opal_common/fetcher/providers/http_fetch_provider.py index 05189876f..aae22a1a4 100644 --- a/packages/opal-common/opal_common/fetcher/providers/http_fetch_provider.py +++ b/packages/opal-common/opal_common/fetcher/providers/http_fetch_provider.py @@ -1,16 +1,18 @@ """Simple HTTP get data fetcher using requests supports.""" from enum import Enum -from typing import Any +from typing import Any, cast +from typing_extensions import Literal +import httpx from aiohttp import ClientResponse, ClientSession -from pydantic import validator +from pydantic import validator, parse_obj_as -from ...http import is_http_error_response -from ...security.sslcontext import get_custom_ssl_context from ..events import FetcherConfig, FetchEvent from ..fetch_provider import BaseFetchProvider from ..logger import get_logger +from ...http import is_http_error_response +from ...security.sslcontext import get_custom_ssl_context logger = get_logger("http_fetch_provider") @@ -48,6 +50,7 @@ class Config: class HttpFetchEvent(FetchEvent): fetcher: str = "HttpFetchProvider" config: HttpFetcherConfig = None + client_type: Literal["httpx", "aiohttp"] = "httpx" class HttpFetchProvider(BaseFetchProvider): @@ -71,13 +74,17 @@ async def __aenter__(self): headers = {} if self._event.config.headers is not None: headers = self._event.config.headers - self._session = await ClientSession( - headers=headers, raise_for_status=True - ).__aenter__() + if self._event.client_type == "httpx": + self._session = httpx.AsyncClient(headers=headers) + else: + self._session = ClientSession( + headers=headers, raise_for_status=True + ) + self._session = self._session.__aenter__() return self async def __aexit__(self, exc_type=None, exc_val=None, tb=None): - await self._session.__aexit__(exc_type=exc_type, exc_val=exc_val, exc_tb=tb) + await self._session.__aexit__(exc_type, exc_val, tb) async def _fetch_(self): logger.debug(f"{self.__class__.__name__} fetching from {self._url}") @@ -96,18 +103,22 @@ async def _fetch_(self): def match_http_method_from_type(session: ClientSession, method_type: HttpMethods): return getattr(session, method_type.value) - async def _process_(self, res: ClientResponse): + @staticmethod + async def _response_to_data(res: ClientResponse | httpx.Response, *, is_json: bool) -> Any: + if isinstance(res,httpx.Response): + return res.json() if is_json else res.text + else: + res = cast(ClientResponse, res) + return await (res.json() if is_json else res.text()) + + async def _process_(self, res: ClientResponse | httpx.Response): # do not process data when the http response is an error if is_http_error_response(res): return res # if we are asked to process the data before we return it if self._event.config.process_data: - # if data is JSON - if self._event.config.is_json: - data = await res.json() - else: - data = await res.text() + data = await self._response_to_data(res, is_json=self._event.config.is_json) return data # return raw result else: diff --git a/packages/opal-common/opal_common/http.py b/packages/opal-common/opal_common/http.py index 8ff942320..7a19cdb5e 100644 --- a/packages/opal-common/opal_common/http.py +++ b/packages/opal-common/opal_common/http.py @@ -1,6 +1,9 @@ import aiohttp +import httpx -def is_http_error_response(response: aiohttp.ClientResponse) -> bool: +def is_http_error_response(response: aiohttp.ClientResponse | httpx.Response) -> bool: """HTTP 400 and above are considered error responses.""" - return response.status >= 400 + status: int = response.status if isinstance(response, aiohttp.ClientResponse) else response.status_code + + return status >= 400 diff --git a/packages/opal-common/requires.txt b/packages/opal-common/requires.txt index 30494e4f0..a40f75b10 100644 --- a/packages/opal-common/requires.txt +++ b/packages/opal-common/requires.txt @@ -10,3 +10,4 @@ datadog>=0.44.0, <1 ddtrace>=2.8.1,<3 certifi>=2023.7.22 # not directly required, pinned by Snyk to avoid a vulnerability requests>=2.31.0 # not directly required, pinned by Snyk to avoid a vulnerability +httpx==0.27.0 \ No newline at end of file