mirror of
				https://github.com/arcan1s/ahriman.git
				synced 2025-10-24 18:33:47 +00:00 
			
		
		
		
	Compare commits
	
		
			5 Commits
		
	
	
		
			e03fcbfab5
			...
			feature/re
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6bc7353514 | |||
| 82d1be52a8 | |||
| 7536d6bb82 | |||
| b050c409cf | |||
| d77cf7c4bb | 
							
								
								
									
										2
									
								
								.github/workflows/setup.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.github/workflows/setup.sh
									
									
									
									
										vendored
									
									
								
							| @ -10,7 +10,7 @@ echo -e '[arcanisrepo]\nServer = http://repo.arcanis.me/$arch\nSigLevel = Never' | |||||||
| # refresh the image | # refresh the image | ||||||
| pacman --noconfirm -Syu | pacman --noconfirm -Syu | ||||||
| # main dependencies | # main dependencies | ||||||
| pacman --noconfirm -Sy base-devel devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo python-systemd sudo | pacman --noconfirm -Sy base-devel devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo python-tenacity sudo | ||||||
| # make dependencies | # make dependencies | ||||||
| pacman --noconfirm -Sy python-build python-flit python-installer python-wheel | pacman --noconfirm -Sy python-build python-flit python-installer python-wheel | ||||||
| # optional dependencies | # optional dependencies | ||||||
|  | |||||||
| @ -122,7 +122,7 @@ Again, the most checks can be performed by `make check` command, though some add | |||||||
|         def __hash__(self) -> int: ...  # basically any magic (or look-alike) method |         def __hash__(self) -> int: ...  # basically any magic (or look-alike) method | ||||||
|     ``` |     ``` | ||||||
|    |    | ||||||
|   Methods inside one group should be ordered alphabetically, the only exceptions are `__init__` (`__post_init__` for dataclasses) and `__new__` methods which should be defined first. For test methods it is recommended to follow the order in which functions are defined. |   Methods inside one group should be ordered alphabetically, the only exceptions are `__init__` (`__post_init__` for dataclasses), `__new__` and `__del__` methods which should be defined first. For test methods it is recommended to follow the order in which functions are defined. | ||||||
|  |  | ||||||
|   Though, we would like to highlight abstract methods (i.e. ones which raise `NotImplementedError`), we still keep in global order at the moment. |   Though, we would like to highlight abstract methods (i.e. ones which raise `NotImplementedError`), we still keep in global order at the moment. | ||||||
|  |  | ||||||
|  | |||||||
| @ -31,7 +31,7 @@ RUN useradd -m -d "/home/build" -s "/usr/bin/nologin" build && \ | |||||||
| COPY "docker/install-aur-package.sh" "/usr/local/bin/install-aur-package" | COPY "docker/install-aur-package.sh" "/usr/local/bin/install-aur-package" | ||||||
| ## install package dependencies | ## install package dependencies | ||||||
| ## darcs is not installed by reasons, because it requires a lot haskell packages which dramatically increase image size | ## darcs is not installed by reasons, because it requires a lot haskell packages which dramatically increase image size | ||||||
| RUN pacman -Sy --noconfirm --asdeps devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo && \ | RUN pacman -Sy --noconfirm --asdeps devtools git pyalpm python-cerberus python-inflection python-passlib python-requests python-srcinfo python-tenacity && \ | ||||||
|     pacman -Sy --noconfirm --asdeps python-build python-flit python-installer python-wheel && \ |     pacman -Sy --noconfirm --asdeps python-build python-flit python-installer python-wheel && \ | ||||||
|     pacman -Sy --noconfirm --asdeps breezy mercurial python-aiohttp python-aiohttp-cors python-boto3 python-cryptography python-jinja python-requests-unixsocket python-systemd rsync subversion && \ |     pacman -Sy --noconfirm --asdeps breezy mercurial python-aiohttp python-aiohttp-cors python-boto3 python-cryptography python-jinja python-requests-unixsocket python-systemd rsync subversion && \ | ||||||
|     runuser -u build -- install-aur-package python-aioauth-client python-aiohttp-apispec-git python-aiohttp-jinja2  \ |     runuser -u build -- install-aur-package python-aioauth-client python-aiohttp-apispec-git python-aiohttp-jinja2  \ | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| Before Width: | Height: | Size: 1.0 MiB After Width: | Height: | Size: 993 KiB | 
| @ -4,6 +4,14 @@ ahriman.core.http package | |||||||
| Submodules | Submodules | ||||||
| ---------- | ---------- | ||||||
|  |  | ||||||
|  | ahriman.core.http.sync\_ahriman\_client module | ||||||
|  | ---------------------------------------------- | ||||||
|  |  | ||||||
|  | .. automodule:: ahriman.core.http.sync_ahriman_client | ||||||
|  |    :members: | ||||||
|  |    :no-undoc-members: | ||||||
|  |    :show-inheritance: | ||||||
|  |  | ||||||
| ahriman.core.http.sync\_http\_client module | ahriman.core.http.sync\_http\_client module | ||||||
| ------------------------------------------- | ------------------------------------------- | ||||||
|  |  | ||||||
|  | |||||||
| @ -111,6 +111,7 @@ Reporting to web service related settings. In most cases there is fallback to we | |||||||
| * ``address`` - remote web service address with protocol, string, optional. In case of websocket, the ``http+unix`` scheme and url encoded address (e.g. ``%2Fvar%2Flib%2Fahriman`` for ``/var/lib/ahriman``) must be used, e.g. ``http+unix://%2Fvar%2Flib%2Fahriman%2Fsocket``. In case if none set, it will be guessed from ``web`` section. | * ``address`` - remote web service address with protocol, string, optional. In case of websocket, the ``http+unix`` scheme and url encoded address (e.g. ``%2Fvar%2Flib%2Fahriman`` for ``/var/lib/ahriman``) must be used, e.g. ``http+unix://%2Fvar%2Flib%2Fahriman%2Fsocket``. In case if none set, it will be guessed from ``web`` section. | ||||||
| * ``password`` - password to authorize in web service in order to update service status, string, required in case if authorization enabled. | * ``password`` - password to authorize in web service in order to update service status, string, required in case if authorization enabled. | ||||||
| * ``suppress_http_log_errors`` - suppress http log errors, boolean, optional, default ``no``. If set to ``yes``, any http log errors (e.g. if web server is not available, but http logging is enabled) will be suppressed. | * ``suppress_http_log_errors`` - suppress http log errors, boolean, optional, default ``no``. If set to ``yes``, any http log errors (e.g. if web server is not available, but http logging is enabled) will be suppressed. | ||||||
|  | * ``timeout`` - HTTP request timeout in seconds, integer, optional, default is ``30``. | ||||||
| * ``username`` - username to authorize in web service in order to update service status, string, required in case if authorization enabled. | * ``username`` - username to authorize in web service in order to update service status, string, required in case if authorization enabled. | ||||||
|  |  | ||||||
| ``web`` group | ``web`` group | ||||||
| @ -129,7 +130,6 @@ Web server settings. If any of ``host``/``port`` is not set, web integration wil | |||||||
| * ``port`` - port to bind, integer, optional. | * ``port`` - port to bind, integer, optional. | ||||||
| * ``static_path`` - path to directory with static files, string, required. | * ``static_path`` - path to directory with static files, string, required. | ||||||
| * ``templates`` - path to templates directories, space separated list of strings, required. | * ``templates`` - path to templates directories, space separated list of strings, required. | ||||||
| * ``timeout`` - HTTP request timeout in seconds, integer, optional, default is ``30``. |  | ||||||
| * ``unix_socket`` - path to the listening unix socket, string, optional. If set, server will create the socket on the specified address which can (and will) be used by application. Note, that unlike usual host/port configuration, unix socket allows to perform requests without authorization. | * ``unix_socket`` - path to the listening unix socket, string, optional. If set, server will create the socket on the specified address which can (and will) be used by application. Note, that unlike usual host/port configuration, unix socket allows to perform requests without authorization. | ||||||
| * ``unix_socket_unsafe`` - set unsafe (o+w) permissions to unix socket, boolean, optional, default ``yes``. This option is enabled by default, because it is supposed that unix socket is created in safe environment (only web service is supposed to be used in unsafe), but it can be disabled by configuration. | * ``unix_socket_unsafe`` - set unsafe (o+w) permissions to unix socket, boolean, optional, default ``yes``. This option is enabled by default, because it is supposed that unix socket is created in safe environment (only web service is supposed to be used in unsafe), but it can be disabled by configuration. | ||||||
| * ``wait_timeout`` - wait timeout in seconds, maximum amount of time to be waited before lock will be free, integer, optional. | * ``wait_timeout`` - wait timeout in seconds, maximum amount of time to be waited before lock will be free, integer, optional. | ||||||
|  | |||||||
| @ -1,13 +1,13 @@ | |||||||
| # Maintainer: Evgeniy Alekseev | # Maintainer: Evgeniy Alekseev | ||||||
|  |  | ||||||
| pkgname='ahriman' | pkgname='ahriman' | ||||||
| pkgver=2.12.1 | pkgver=2.12.2 | ||||||
| pkgrel=1 | pkgrel=1 | ||||||
| pkgdesc="ArcH linux ReposItory MANager" | pkgdesc="ArcH linux ReposItory MANager" | ||||||
| arch=('any') | arch=('any') | ||||||
| url="https://github.com/arcan1s/ahriman" | url="https://github.com/arcan1s/ahriman" | ||||||
| license=('GPL3') | license=('GPL3') | ||||||
| depends=('devtools>=1:1.0.0' 'git' 'pyalpm' 'python-cerberus' 'python-inflection' 'python-passlib' 'python-requests' 'python-srcinfo') | depends=('devtools>=1:1.0.0' 'git' 'pyalpm' 'python-cerberus' 'python-inflection' 'python-passlib' 'python-requests' 'python-srcinfo' 'python-tenacity') | ||||||
| makedepends=('python-build' 'python-flit' 'python-installer' 'python-wheel') | makedepends=('python-build' 'python-flit' 'python-installer' 'python-wheel') | ||||||
| optdepends=('breezy: -bzr packages support' | optdepends=('breezy: -bzr packages support' | ||||||
|             'darcs: -darcs packages support' |             'darcs: -darcs packages support' | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| .TH AHRIMAN "1" "2023\-11\-12" "ahriman" "Generated Python Manual" | .TH AHRIMAN "1" "2023\-11\-13" "ahriman" "Generated Python Manual" | ||||||
| .SH NAME | .SH NAME | ||||||
| ahriman | ahriman | ||||||
| .SH SYNOPSIS | .SH SYNOPSIS | ||||||
|  | |||||||
| @ -31,6 +31,7 @@ class MethodTypeOrder(StrEnum): | |||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         Class(MethodTypeOrder): (class attribute) class method |         Class(MethodTypeOrder): (class attribute) class method | ||||||
|  |         Delete(MethodTypeOrder): (class attribute) destructor-like methods | ||||||
|         Init(MethodTypeOrder): (class attribute) initialization method |         Init(MethodTypeOrder): (class attribute) initialization method | ||||||
|         Magic(MethodTypeOrder): (class attribute) other magical methods |         Magic(MethodTypeOrder): (class attribute) other magical methods | ||||||
|         New(MethodTypeOrder): (class attribute) constructor method |         New(MethodTypeOrder): (class attribute) constructor method | ||||||
| @ -40,6 +41,7 @@ class MethodTypeOrder(StrEnum): | |||||||
|     """ |     """ | ||||||
|  |  | ||||||
|     Class = "classmethod" |     Class = "classmethod" | ||||||
|  |     Delete = "del" | ||||||
|     Init = "init" |     Init = "init" | ||||||
|     Magic = "magic" |     Magic = "magic" | ||||||
|     New = "new" |     New = "new" | ||||||
| @ -76,8 +78,9 @@ class DefinitionOrder(BaseRawFileChecker): | |||||||
|             "method-type-order", |             "method-type-order", | ||||||
|             { |             { | ||||||
|                 "default": [ |                 "default": [ | ||||||
|                     "new", |  | ||||||
|                     "init", |                     "init", | ||||||
|  |                     "new", | ||||||
|  |                     "del", | ||||||
|                     "property", |                     "property", | ||||||
|                     "classmethod", |                     "classmethod", | ||||||
|                     "staticmethod", |                     "staticmethod", | ||||||
| @ -122,10 +125,12 @@ class DefinitionOrder(BaseRawFileChecker): | |||||||
|             MethodTypeOrder: resolved function type |             MethodTypeOrder: resolved function type | ||||||
|         """ |         """ | ||||||
|         # init methods |         # init methods | ||||||
|         if function.name in ("__new__",): |  | ||||||
|             return MethodTypeOrder.New |  | ||||||
|         if function.name in ("__init__", "__post_init__"): |         if function.name in ("__init__", "__post_init__"): | ||||||
|             return MethodTypeOrder.Init |             return MethodTypeOrder.Init | ||||||
|  |         if function.name in ("__new__",): | ||||||
|  |             return MethodTypeOrder.New | ||||||
|  |         if function.name in ("__del__",): | ||||||
|  |             return MethodTypeOrder.Delete | ||||||
|  |  | ||||||
|         # decorated methods |         # decorated methods | ||||||
|         decorators = [] |         decorators = [] | ||||||
|  | |||||||
| @ -21,6 +21,7 @@ dependencies = [ | |||||||
|     "passlib", |     "passlib", | ||||||
|     "requests", |     "requests", | ||||||
|     "srcinfo", |     "srcinfo", | ||||||
|  |     "tenacity", | ||||||
| ] | ] | ||||||
|  |  | ||||||
| dynamic = ["version"] | dynamic = ["version"] | ||||||
|  | |||||||
| @ -17,4 +17,4 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
| __version__ = "2.12.1" | __version__ = "2.12.2" | ||||||
|  | |||||||
| @ -269,6 +269,11 @@ CONFIGURATION_SCHEMA: ConfigurationSchema = { | |||||||
|                 "type": "boolean", |                 "type": "boolean", | ||||||
|                 "coerce": "boolean", |                 "coerce": "boolean", | ||||||
|             }, |             }, | ||||||
|  |             "timeout": { | ||||||
|  |                 "type": "integer", | ||||||
|  |                 "coerce": "integer", | ||||||
|  |                 "min": 0, | ||||||
|  |             }, | ||||||
|             "username": { |             "username": { | ||||||
|                 "type": "string", |                 "type": "string", | ||||||
|                 "empty": False, |                 "empty": False, | ||||||
|  | |||||||
| @ -17,4 +17,5 @@ | |||||||
| # You should have received a copy of the GNU General Public License | # You should have received a copy of the GNU General Public License | ||||||
| # along with this program. If not, see <http://www.gnu.org/licenses/>. | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
| # | # | ||||||
|  | from ahriman.core.http.sync_ahriman_client import SyncAhrimanClient | ||||||
| from ahriman.core.http.sync_http_client import MultipartType, SyncHttpClient | from ahriman.core.http.sync_http_client import MultipartType, SyncHttpClient | ||||||
|  | |||||||
							
								
								
									
										134
									
								
								src/ahriman/core/http/sync_ahriman_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										134
									
								
								src/ahriman/core/http/sync_ahriman_client.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,134 @@ | |||||||
|  | # | ||||||
|  | # Copyright (c) 2021-2023 ahriman team. | ||||||
|  | # | ||||||
|  | # This file is part of ahriman | ||||||
|  | # (see https://github.com/arcan1s/ahriman). | ||||||
|  | # | ||||||
|  | # This program is free software: you can redistribute it and/or modify | ||||||
|  | # it under the terms of the GNU General Public License as published by | ||||||
|  | # the Free Software Foundation, either version 3 of the License, or | ||||||
|  | # (at your option) any later version. | ||||||
|  | # | ||||||
|  | # This program is distributed in the hope that it will be useful, | ||||||
|  | # but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||||
|  | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||||
|  | # GNU General Public License for more details. | ||||||
|  | # | ||||||
|  | # You should have received a copy of the GNU General Public License | ||||||
|  | # along with this program. If not, see <http://www.gnu.org/licenses/>. | ||||||
|  | # | ||||||
|  | import contextlib | ||||||
|  | import requests | ||||||
|  | import tenacity | ||||||
|  |  | ||||||
|  | from functools import cached_property | ||||||
|  | from typing import Any | ||||||
|  | from urllib.parse import urlparse | ||||||
|  |  | ||||||
|  | from ahriman import __version__ | ||||||
|  | from ahriman.core.http.sync_http_client import SyncHttpClient | ||||||
|  |  | ||||||
|  |  | ||||||
|  | class SyncAhrimanClient(SyncHttpClient): | ||||||
|  |     """ | ||||||
|  |     wrapper for ahriman web service | ||||||
|  |  | ||||||
|  |     Attributes: | ||||||
|  |         address(str): address of the web service | ||||||
|  |     """ | ||||||
|  |  | ||||||
|  |     address: str | ||||||
|  |  | ||||||
|  |     @cached_property | ||||||
|  |     def session(self) -> requests.Session: | ||||||
|  |         """ | ||||||
|  |         get or create session | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             request.Session: created session object | ||||||
|  |         """ | ||||||
|  |         if urlparse(self.address).scheme == "http+unix": | ||||||
|  |             import requests_unixsocket  # type: ignore[import-untyped] | ||||||
|  |             session: requests.Session = requests_unixsocket.Session() | ||||||
|  |             session.headers["User-Agent"] = f"ahriman/{__version__}" | ||||||
|  |             return session | ||||||
|  |  | ||||||
|  |         session = requests.Session() | ||||||
|  |         session.headers["User-Agent"] = f"ahriman/{__version__}" | ||||||
|  |         self._login(session) | ||||||
|  |  | ||||||
|  |         return session | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def is_retry_allowed(exception: BaseException) -> bool: | ||||||
|  |         """ | ||||||
|  |         check if retry is allowed for the exception | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             exception(BaseException): exception raised | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             bool: True in case if exception is in white list and false otherwise | ||||||
|  |         """ | ||||||
|  |         if not isinstance(exception, requests.RequestException): | ||||||
|  |             return False  # not a request exception | ||||||
|  |         status_code = exception.response.status_code if exception.response is not None else None | ||||||
|  |         return status_code in (401,) | ||||||
|  |  | ||||||
|  |     @staticmethod | ||||||
|  |     def on_retry(state: tenacity.RetryCallState) -> None: | ||||||
|  |         """ | ||||||
|  |         action to be called before retry | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             state(tenacity.RetryCallState): current retry call state | ||||||
|  |         """ | ||||||
|  |         instance = next(arg for arg in state.args if isinstance(arg, SyncAhrimanClient)) | ||||||
|  |         if hasattr(instance, "session"):  # only if it was initialized | ||||||
|  |             del instance.session  # clear session | ||||||
|  |  | ||||||
|  |     def _login(self, session: requests.Session) -> None: | ||||||
|  |         """ | ||||||
|  |         process login to the service | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             session(requests.Session): request session to login | ||||||
|  |         """ | ||||||
|  |         if self.auth is None: | ||||||
|  |             return  # no auth configured | ||||||
|  |  | ||||||
|  |         username, password = self.auth | ||||||
|  |         payload = { | ||||||
|  |             "username": username, | ||||||
|  |             "password": password, | ||||||
|  |         } | ||||||
|  |         with contextlib.suppress(Exception): | ||||||
|  |             self.make_request("POST", self._login_url(), json=payload, session=session) | ||||||
|  |  | ||||||
|  |     def _login_url(self) -> str: | ||||||
|  |         """ | ||||||
|  |         get url for the login api | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             str: full url for web service to log in | ||||||
|  |         """ | ||||||
|  |         return f"{self.address}/api/v1/login" | ||||||
|  |  | ||||||
|  |     @tenacity.retry( | ||||||
|  |         stop=tenacity.stop_after_attempt(2), | ||||||
|  |         retry=tenacity.retry_if_exception(is_retry_allowed), | ||||||
|  |         after=on_retry, | ||||||
|  |         reraise=True, | ||||||
|  |     ) | ||||||
|  |     def make_request(self, *args: Any, **kwargs: Any) -> requests.Response: | ||||||
|  |         """ | ||||||
|  |         perform request with specified parameters | ||||||
|  |  | ||||||
|  |         Args: | ||||||
|  |             *args(Any): request method positional arguments | ||||||
|  |             **kwargs(Any): request method keyword arguments | ||||||
|  |  | ||||||
|  |         Returns: | ||||||
|  |             requests.Response: response object | ||||||
|  |         """ | ||||||
|  |         return SyncHttpClient.make_request(self, *args, **kwargs) | ||||||
| @ -77,12 +77,12 @@ class SyncHttpClient(LazyLogging): | |||||||
|         return session |         return session | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def exception_response_text(exception: requests.exceptions.RequestException) -> str: |     def exception_response_text(exception: requests.RequestException) -> str: | ||||||
|         """ |         """ | ||||||
|         safe response exception text generation |         safe response exception text generation | ||||||
|  |  | ||||||
|         Args: |         Args: | ||||||
|             exception(requests.exceptions.RequestException): exception raised |             exception(requests.RequestException): exception raised | ||||||
|  |  | ||||||
|         Returns: |         Returns: | ||||||
|             str: text of the response if it is not None and empty string otherwise |             str: text of the response if it is not None and empty string otherwise | ||||||
|  | |||||||
| @ -19,14 +19,11 @@ | |||||||
| # | # | ||||||
| import contextlib | import contextlib | ||||||
| import logging | import logging | ||||||
| import requests |  | ||||||
|  |  | ||||||
| from functools import cached_property | from urllib.parse import quote_plus as urlencode | ||||||
| from urllib.parse import quote_plus as urlencode, urlparse |  | ||||||
|  |  | ||||||
| from ahriman import __version__ |  | ||||||
| from ahriman.core.configuration import Configuration | from ahriman.core.configuration import Configuration | ||||||
| from ahriman.core.http import SyncHttpClient | from ahriman.core.http import SyncAhrimanClient | ||||||
| from ahriman.core.status.client import Client | from ahriman.core.status.client import Client | ||||||
| from ahriman.models.build_status import BuildStatus, BuildStatusEnum | from ahriman.models.build_status import BuildStatus, BuildStatusEnum | ||||||
| from ahriman.models.internal_status import InternalStatus | from ahriman.models.internal_status import InternalStatus | ||||||
| @ -35,12 +32,11 @@ from ahriman.models.package import Package | |||||||
| from ahriman.models.repository_id import RepositoryId | from ahriman.models.repository_id import RepositoryId | ||||||
|  |  | ||||||
|  |  | ||||||
| class WebClient(Client, SyncHttpClient): | class WebClient(Client, SyncAhrimanClient): | ||||||
|     """ |     """ | ||||||
|     build status reporter web client |     build status reporter web client | ||||||
|  |  | ||||||
|     Attributes: |     Attributes: | ||||||
|         address(str): address of the web service |  | ||||||
|         repository_id(RepositoryId): repository unique identifier |         repository_id(RepositoryId): repository unique identifier | ||||||
|     """ |     """ | ||||||
|  |  | ||||||
| @ -56,30 +52,10 @@ class WebClient(Client, SyncHttpClient): | |||||||
|         suppress_errors = configuration.getboolean(  # read old-style first and then fallback to new style |         suppress_errors = configuration.getboolean(  # read old-style first and then fallback to new style | ||||||
|             "settings", "suppress_http_log_errors", |             "settings", "suppress_http_log_errors", | ||||||
|             fallback=configuration.getboolean("status", "suppress_http_log_errors", fallback=False)) |             fallback=configuration.getboolean("status", "suppress_http_log_errors", fallback=False)) | ||||||
|         SyncHttpClient.__init__(self, configuration, section, suppress_errors=suppress_errors) |         SyncAhrimanClient.__init__(self, configuration, section, suppress_errors=suppress_errors) | ||||||
|  |  | ||||||
|         self.repository_id = repository_id |         self.repository_id = repository_id | ||||||
|  |  | ||||||
|     @cached_property |  | ||||||
|     def session(self) -> requests.Session: |  | ||||||
|         """ |  | ||||||
|         get or create session |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             request.Session: created session object |  | ||||||
|         """ |  | ||||||
|         if urlparse(self.address).scheme == "http+unix": |  | ||||||
|             import requests_unixsocket  # type: ignore[import-untyped] |  | ||||||
|             session: requests.Session = requests_unixsocket.Session() |  | ||||||
|             session.headers["User-Agent"] = f"ahriman/{__version__}" |  | ||||||
|             return session |  | ||||||
|  |  | ||||||
|         session = requests.Session() |  | ||||||
|         session.headers["User-Agent"] = f"ahriman/{__version__}" |  | ||||||
|         self._login(session) |  | ||||||
|  |  | ||||||
|         return session |  | ||||||
|  |  | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def parse_address(configuration: Configuration) -> tuple[str, str]: |     def parse_address(configuration: Configuration) -> tuple[str, str]: | ||||||
|         """ |         """ | ||||||
| @ -107,33 +83,6 @@ class WebClient(Client, SyncHttpClient): | |||||||
|             address = f"http://{host}:{port}" |             address = f"http://{host}:{port}" | ||||||
|         return "web", address |         return "web", address | ||||||
|  |  | ||||||
|     def _login(self, session: requests.Session) -> None: |  | ||||||
|         """ |  | ||||||
|         process login to the service |  | ||||||
|  |  | ||||||
|         Args: |  | ||||||
|             session(requests.Session): request session to login |  | ||||||
|         """ |  | ||||||
|         if self.auth is None: |  | ||||||
|             return  # no auth configured |  | ||||||
|  |  | ||||||
|         username, password = self.auth |  | ||||||
|         payload = { |  | ||||||
|             "username": username, |  | ||||||
|             "password": password, |  | ||||||
|         } |  | ||||||
|         with contextlib.suppress(Exception): |  | ||||||
|             self.make_request("POST", self._login_url(), json=payload, session=session) |  | ||||||
|  |  | ||||||
|     def _login_url(self) -> str: |  | ||||||
|         """ |  | ||||||
|         get url for the login api |  | ||||||
|  |  | ||||||
|         Returns: |  | ||||||
|             str: full url for web service to log in |  | ||||||
|         """ |  | ||||||
|         return f"{self.address}/api/v1/login" |  | ||||||
|  |  | ||||||
|     def _logs_url(self, package_base: str) -> str: |     def _logs_url(self, package_base: str) -> str: | ||||||
|         """ |         """ | ||||||
|         get url for the logs api |         get url for the logs api | ||||||
|  | |||||||
| @ -65,6 +65,14 @@ class TriggerLoader(LazyLogging): | |||||||
|         self._on_stop_requested = False |         self._on_stop_requested = False | ||||||
|         self.triggers: list[Trigger] = [] |         self.triggers: list[Trigger] = [] | ||||||
|  |  | ||||||
|  |     def __del__(self) -> None: | ||||||
|  |         """ | ||||||
|  |         custom destructor object which calls on_stop in case if it was requested | ||||||
|  |         """ | ||||||
|  |         if not self._on_stop_requested: | ||||||
|  |             return | ||||||
|  |         self.on_stop() | ||||||
|  |  | ||||||
|     @classmethod |     @classmethod | ||||||
|     def load(cls, repository_id: RepositoryId, configuration: Configuration) -> Self: |     def load(cls, repository_id: RepositoryId, configuration: Configuration) -> Self: | ||||||
|         """ |         """ | ||||||
| @ -257,11 +265,3 @@ class TriggerLoader(LazyLogging): | |||||||
|         for trigger in self.triggers: |         for trigger in self.triggers: | ||||||
|             with self.__execute_trigger(trigger): |             with self.__execute_trigger(trigger): | ||||||
|                 trigger.on_stop() |                 trigger.on_stop() | ||||||
|  |  | ||||||
|     def __del__(self) -> None: |  | ||||||
|         """ |  | ||||||
|         custom destructor object which calls on_stop in case if it was requested |  | ||||||
|         """ |  | ||||||
|         if not self._on_stop_requested: |  | ||||||
|             return |  | ||||||
|         self.on_stop() |  | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								tests/ahriman/core/http/conftest.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								tests/ahriman/core/http/conftest.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | import pytest | ||||||
|  |  | ||||||
|  | from ahriman.core.configuration import Configuration | ||||||
|  | from ahriman.core.http import SyncAhrimanClient | ||||||
|  | from ahriman.core.status.web_client import WebClient | ||||||
|  |  | ||||||
|  |  | ||||||
|  | @pytest.fixture | ||||||
|  | def ahriman_client(configuration: Configuration) -> SyncAhrimanClient: | ||||||
|  |     """ | ||||||
|  |     ahriman web client fixture | ||||||
|  |  | ||||||
|  |     Args: | ||||||
|  |         configuration(Configuration): configuration fixture | ||||||
|  |  | ||||||
|  |     Returns: | ||||||
|  |         SyncAhrimanClient: ahriman web client test instance | ||||||
|  |     """ | ||||||
|  |     configuration.set("web", "port", "8080") | ||||||
|  |     _, repository_id = configuration.check_loaded() | ||||||
|  |     return WebClient(repository_id, configuration) | ||||||
							
								
								
									
										146
									
								
								tests/ahriman/core/http/test_sync_ahriman_client.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										146
									
								
								tests/ahriman/core/http/test_sync_ahriman_client.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,146 @@ | |||||||
|  | import pytest | ||||||
|  | import requests | ||||||
|  | import requests_unixsocket | ||||||
|  | import tenacity | ||||||
|  |  | ||||||
|  | from pytest_mock import MockerFixture | ||||||
|  | from unittest.mock import call as MockCall | ||||||
|  |  | ||||||
|  | from ahriman.core.http import SyncAhrimanClient | ||||||
|  | from ahriman.models.user import User | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_session(ahriman_client: SyncAhrimanClient, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must create normal requests session | ||||||
|  |     """ | ||||||
|  |     login_mock = mocker.patch("ahriman.core.http.SyncAhrimanClient._login") | ||||||
|  |  | ||||||
|  |     assert isinstance(ahriman_client.session, requests.Session) | ||||||
|  |     assert not isinstance(ahriman_client.session, requests_unixsocket.Session) | ||||||
|  |     login_mock.assert_called_once_with(pytest.helpers.anyvar(int)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_session_unix_socket(ahriman_client: SyncAhrimanClient, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must create unix socket session | ||||||
|  |     """ | ||||||
|  |     login_mock = mocker.patch("ahriman.core.http.SyncAhrimanClient._login") | ||||||
|  |     ahriman_client.address = "http+unix://path" | ||||||
|  |  | ||||||
|  |     assert isinstance(ahriman_client.session, requests_unixsocket.Session) | ||||||
|  |     login_mock.assert_not_called() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_is_retry_allowed() -> None: | ||||||
|  |     """ | ||||||
|  |     must allow retries on 401 errors | ||||||
|  |     """ | ||||||
|  |     assert not SyncAhrimanClient.is_retry_allowed(Exception()) | ||||||
|  |  | ||||||
|  |     response = requests.Response() | ||||||
|  |     response.status_code = 400 | ||||||
|  |     assert not SyncAhrimanClient.is_retry_allowed(requests.HTTPError(response=response)) | ||||||
|  |  | ||||||
|  |     response.status_code = 401 | ||||||
|  |     assert SyncAhrimanClient.is_retry_allowed(requests.HTTPError(response=response)) | ||||||
|  |  | ||||||
|  |     response.status_code = 403 | ||||||
|  |     assert not SyncAhrimanClient.is_retry_allowed(requests.HTTPError(response=response)) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_on_retry(ahriman_client: SyncAhrimanClient) -> None: | ||||||
|  |     """ | ||||||
|  |     must remove session on retry | ||||||
|  |     """ | ||||||
|  |     SyncAhrimanClient.on_retry( | ||||||
|  |         tenacity.RetryCallState( | ||||||
|  |             retry_object=tenacity.Retrying(), | ||||||
|  |             fn=SyncAhrimanClient.make_request, | ||||||
|  |             args=(ahriman_client,), | ||||||
|  |             kwargs={}, | ||||||
|  |         ) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_login(ahriman_client: SyncAhrimanClient, user: User, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must login user | ||||||
|  |     """ | ||||||
|  |     ahriman_client.auth = (user.username, user.password) | ||||||
|  |     requests_mock = mocker.patch("ahriman.core.http.SyncAhrimanClient.make_request") | ||||||
|  |     payload = { | ||||||
|  |         "username": user.username, | ||||||
|  |         "password": user.password | ||||||
|  |     } | ||||||
|  |     session = requests.Session() | ||||||
|  |  | ||||||
|  |     ahriman_client._login(session) | ||||||
|  |     requests_mock.assert_called_once_with("POST", pytest.helpers.anyvar(str, True), json=payload, session=session) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_login_failed(ahriman_client: SyncAhrimanClient, user: User, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must suppress any exception happened during login | ||||||
|  |     """ | ||||||
|  |     ahriman_client.user = user | ||||||
|  |     mocker.patch("requests.Session.request", side_effect=Exception()) | ||||||
|  |     ahriman_client._login(requests.Session()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_login_failed_http_error(ahriman_client: SyncAhrimanClient, user: User, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must suppress HTTP exception happened during login | ||||||
|  |     """ | ||||||
|  |     ahriman_client.user = user | ||||||
|  |     mocker.patch("requests.Session.request", side_effect=requests.HTTPError()) | ||||||
|  |     ahriman_client._login(requests.Session()) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_login_skip(ahriman_client: SyncAhrimanClient, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must skip login if no user set | ||||||
|  |     """ | ||||||
|  |     requests_mock = mocker.patch("requests.Session.request") | ||||||
|  |     ahriman_client._login(requests.Session()) | ||||||
|  |     requests_mock.assert_not_called() | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_login_url(ahriman_client: SyncAhrimanClient) -> None: | ||||||
|  |     """ | ||||||
|  |     must generate login url correctly | ||||||
|  |     """ | ||||||
|  |     assert ahriman_client._login_url().startswith(ahriman_client.address) | ||||||
|  |     assert ahriman_client._login_url().endswith("/api/v1/login") | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def test_make_request_retry(ahriman_client: SyncAhrimanClient, mocker: MockerFixture) -> None: | ||||||
|  |     """ | ||||||
|  |     must retry HTTP request | ||||||
|  |     """ | ||||||
|  |     response_ok = requests.Response() | ||||||
|  |     response_ok.status_code = 200 | ||||||
|  |     response_error = requests.Response() | ||||||
|  |     response_error.status_code = 401 | ||||||
|  |     # login on init -> request with error -> login on error -> request without error | ||||||
|  |     request_mock = mocker.patch("requests.Session.request", side_effect=[ | ||||||
|  |         response_ok, | ||||||
|  |         response_error, | ||||||
|  |         response_ok, | ||||||
|  |         response_ok, | ||||||
|  |     ]) | ||||||
|  |  | ||||||
|  |     ahriman_client.auth = ("username", "password") | ||||||
|  |     assert ahriman_client.make_request("GET", "url") is not None | ||||||
|  |     request_mock.assert_has_calls([ | ||||||
|  |         MockCall("POST", "http://127.0.0.1:8080/api/v1/login", params=None, data=None, headers=None, files=None, | ||||||
|  |                  json={"username": "username", "password": "password"}, | ||||||
|  |                  auth=ahriman_client.auth, timeout=ahriman_client.timeout), | ||||||
|  |         MockCall("GET", "url", params=None, data=None, headers=None, files=None, json=None, | ||||||
|  |                  auth=ahriman_client.auth, timeout=ahriman_client.timeout), | ||||||
|  |         MockCall("POST", "http://127.0.0.1:8080/api/v1/login", params=None, data=None, headers=None, files=None, | ||||||
|  |                  json={"username": "username", "password": "password"}, | ||||||
|  |                  auth=ahriman_client.auth, timeout=ahriman_client.timeout), | ||||||
|  |         MockCall("GET", "url", params=None, data=None, headers=None, files=None, json=None, | ||||||
|  |                  auth=ahriman_client.auth, timeout=ahriman_client.timeout), | ||||||
|  |     ]) | ||||||
| @ -2,7 +2,6 @@ import json | |||||||
| import logging | import logging | ||||||
| import pytest | import pytest | ||||||
| import requests | import requests | ||||||
| import requests_unixsocket |  | ||||||
|  |  | ||||||
| from pytest_mock import MockerFixture | from pytest_mock import MockerFixture | ||||||
|  |  | ||||||
| @ -12,29 +11,6 @@ from ahriman.models.build_status import BuildStatus, BuildStatusEnum | |||||||
| from ahriman.models.internal_status import InternalStatus | from ahriman.models.internal_status import InternalStatus | ||||||
| from ahriman.models.log_record_id import LogRecordId | from ahriman.models.log_record_id import LogRecordId | ||||||
| from ahriman.models.package import Package | from ahriman.models.package import Package | ||||||
| from ahriman.models.user import User |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_session(web_client: WebClient, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must create normal requests session |  | ||||||
|     """ |  | ||||||
|     login_mock = mocker.patch("ahriman.core.status.web_client.WebClient._login") |  | ||||||
|  |  | ||||||
|     assert isinstance(web_client.session, requests.Session) |  | ||||||
|     assert not isinstance(web_client.session, requests_unixsocket.Session) |  | ||||||
|     login_mock.assert_called_once_with(pytest.helpers.anyvar(int)) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_session_unix_socket(web_client: WebClient, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must create unix socket session |  | ||||||
|     """ |  | ||||||
|     login_mock = mocker.patch("ahriman.core.status.web_client.WebClient._login") |  | ||||||
|     web_client.address = "http+unix://path" |  | ||||||
|  |  | ||||||
|     assert isinstance(web_client.session, requests_unixsocket.Session) |  | ||||||
|     login_mock.assert_not_called() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_parse_address(configuration: Configuration) -> None: | def test_parse_address(configuration: Configuration) -> None: | ||||||
| @ -55,57 +31,6 @@ def test_parse_address(configuration: Configuration) -> None: | |||||||
|     assert WebClient.parse_address(configuration) == ("status", "http://localhost:8082") |     assert WebClient.parse_address(configuration) == ("status", "http://localhost:8082") | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_login(web_client: WebClient, user: User, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must login user |  | ||||||
|     """ |  | ||||||
|     web_client.auth = (user.username, user.password) |  | ||||||
|     requests_mock = mocker.patch("ahriman.core.status.web_client.WebClient.make_request") |  | ||||||
|     payload = { |  | ||||||
|         "username": user.username, |  | ||||||
|         "password": user.password |  | ||||||
|     } |  | ||||||
|     session = requests.Session() |  | ||||||
|  |  | ||||||
|     web_client._login(session) |  | ||||||
|     requests_mock.assert_called_once_with("POST", pytest.helpers.anyvar(str, True), json=payload, session=session) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_login_failed(web_client: WebClient, user: User, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must suppress any exception happened during login |  | ||||||
|     """ |  | ||||||
|     web_client.user = user |  | ||||||
|     mocker.patch("requests.Session.request", side_effect=Exception()) |  | ||||||
|     web_client._login(requests.Session()) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_login_failed_http_error(web_client: WebClient, user: User, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must suppress HTTP exception happened during login |  | ||||||
|     """ |  | ||||||
|     web_client.user = user |  | ||||||
|     mocker.patch("requests.Session.request", side_effect=requests.HTTPError()) |  | ||||||
|     web_client._login(requests.Session()) |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_login_skip(web_client: WebClient, mocker: MockerFixture) -> None: |  | ||||||
|     """ |  | ||||||
|     must skip login if no user set |  | ||||||
|     """ |  | ||||||
|     requests_mock = mocker.patch("requests.Session.request") |  | ||||||
|     web_client._login(requests.Session()) |  | ||||||
|     requests_mock.assert_not_called() |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_login_url(web_client: WebClient) -> None: |  | ||||||
|     """ |  | ||||||
|     must generate login url correctly |  | ||||||
|     """ |  | ||||||
|     assert web_client._login_url().startswith(web_client.address) |  | ||||||
|     assert web_client._login_url().endswith("/api/v1/login") |  | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_status_url(web_client: WebClient) -> None: | def test_status_url(web_client: WebClient) -> None: | ||||||
|     """ |     """ | ||||||
|     must generate package status url correctly |     must generate package status url correctly | ||||||
| @ -377,11 +302,13 @@ def test_status_update(web_client: WebClient, mocker: MockerFixture) -> None: | |||||||
|     requests_mock = mocker.patch("ahriman.core.status.web_client.WebClient.make_request") |     requests_mock = mocker.patch("ahriman.core.status.web_client.WebClient.make_request") | ||||||
|  |  | ||||||
|     web_client.status_update(BuildStatusEnum.Unknown) |     web_client.status_update(BuildStatusEnum.Unknown) | ||||||
|     requests_mock.assert_called_once_with("POST", pytest.helpers.anyvar(str, True), |     requests_mock.assert_called_once_with( | ||||||
|                                           params=web_client.repository_id.query(), |         "POST", pytest.helpers.anyvar(str, True), | ||||||
|                                           json={ |         params=web_client.repository_id.query(), | ||||||
|                                               "status": BuildStatusEnum.Unknown.value, |         json={ | ||||||
|     }) |             "status": BuildStatusEnum.Unknown.value, | ||||||
|  |         } | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
| def test_status_update_self_failed(web_client: WebClient, mocker: MockerFixture) -> None: | def test_status_update_self_failed(web_client: WebClient, mocker: MockerFixture) -> None: | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user