Remnawave_python-sdk/remnawave/__init__.py
Artem 0ed1ce92c0
feat: Introduce metadata management for users and nodes
- Added MetadataController for handling user and node metadata.
- Implemented models for user and node metadata management.
- Created tests for user and node metadata functionalities.
- Enhanced authentication settings with passkey and OAuth2 configurations.
- Added bulk actions for node updates and responses.
- Refactored existing models to accommodate new features and improve structure.
- Removed obsolete test_imports.py file.
- Updated environment variables for testing.
- Improved error handling in subscription tests.
- Added new node plugin functionalities including cloning and execution commands.
2026-03-11 12:35:42 +01:00

165 lines
6.2 KiB
Python

import logging
from typing import Optional
import httpx
from remnawave.controllers import (
APITokensManagementController,
AuthController,
BandWidthStatsController,
ConfigProfilesController,
HostsBulkActionsController,
HostsController,
HWIDUserController,
InboundsBulkActionsController,
InboundsController,
InfraBillingController,
InternalSquadsController,
KeygenController,
NodesController,
SubscriptionController,
SubscriptionsController,
SubscriptionsSettingsController,
SubscriptionsTemplateController,
SystemController,
UsersBulkActionsController,
UsersController,
WebhookUtility,
XrayConfigController,
SubscriptionRequestHistoryController,
PasskeysController,
ExternalSquadsController,
SnippetsController,
RemnawaveSettingsController,
SubscriptionPageConfigController,
IpControlController,
NodePluginsController,
MetadataController,
)
class RemnawaveSDK:
def __init__(
self,
client: Optional[httpx.AsyncClient] = None,
base_url: Optional[str] = None,
token: Optional[str] = None,
caddy_token: Optional[str] = None,
ssl_ignore: Optional[bool] = False,
custom_headers: Optional[dict] = None,
cookies: Optional[dict] = None,
):
"""
Remnawave SDK init
Args:
client (Optional[httpx.AsyncClient]): - Default client.
base_url (Optional[str]): - Base url of the Remnawave panel. Defaults to None.
token (Optional[str]): - Token for authorization.
caddy_token (Optional[str]): - Token for Caddy Auth (Headers). Defaults to None.
ssl_ignore (Optional[bool]): - Whether to ignore SSL certificate errors. Defaults to False.
custom_headers (Optional[dict]): - Custom headers to include in the requests. Defaults to None.
cookies (Optional[dict]): - Cookies for Reverse Proxy authorization. Defaults to None.
"""
self._client = client
self._token = token
self.base_url = base_url
self.caddy_token = caddy_token
self.ssl_ignore = ssl_ignore
self.custom_headers = custom_headers
self.cookies = cookies
self._validate_params()
if self._client is None:
self._client = self._prepare_client()
self.api_tokens_management = APITokensManagementController(self._client)
self.auth = AuthController(self._client)
self.bandwidthstats = BandWidthStatsController(self._client)
self.config_profiles = ConfigProfilesController(self._client)
self.hosts = HostsController(self._client)
self.hosts_bulk_actions = HostsBulkActionsController(self._client)
self.hwid = HWIDUserController(self._client)
self.inbounds = InboundsController(self._client)
self.inbounds_bulk_actions = InboundsBulkActionsController(self._client)
self.infra_billing = InfraBillingController(self._client)
self.internal_squads = InternalSquadsController(self._client)
self.keygen = KeygenController(self._client)
self.nodes = NodesController(self._client)
self.subscription = SubscriptionController(self._client)
self.subscriptions = SubscriptionsController(self._client)
self.subscriptions_settings = SubscriptionsSettingsController(self._client)
self.subscriptions_template = SubscriptionsTemplateController(self._client)
self.subscription_request_history = SubscriptionRequestHistoryController(self._client)
self.system = SystemController(self._client)
self.users = UsersController(self._client)
self.users_bulk_actions = UsersBulkActionsController(self._client)
self.webhook_utility = WebhookUtility()
self.xray_config = XrayConfigController(self._client)
self.passkeys = PasskeysController(self._client)
self.external_squads = ExternalSquadsController(self._client)
self.snippets = SnippetsController(self._client)
self.remnawave_settings = RemnawaveSettingsController(self._client)
self.subscription_page_config = SubscriptionPageConfigController(self._client)
self.ip_control = IpControlController(self._client)
self.node_plugins = NodePluginsController(self._client)
self.metadata = MetadataController(self._client)
def _validate_params(self) -> None:
if self._client is None:
if self.base_url is None or self._token is None:
raise ValueError(
"base_url and token must be provided if client is not provided"
)
else:
if self.base_url is not None or self._token is not None:
logging.warning(
"base_url and token will be ignored if client is provided"
)
if self.cookies is not None:
logging.warning(
"cookies will be ignored if client is provided"
)
def _prepare_client(self) -> httpx.AsyncClient:
client_kwargs = {
"base_url": self._prepare_url(),
"headers": self._prepare_headers(),
"verify": not self.ssl_ignore,
}
if self.cookies is not None:
client_kwargs["cookies"] = self.cookies
return httpx.AsyncClient(**client_kwargs)
def _prepare_headers(self) -> dict:
headers = {}
if self._token:
headers["Authorization"] = (
self._token
if self._token.startswith("Bearer ")
else f"Bearer {self._token}"
)
# X-Api-Key for Caddy (https://remna.st/security/caddy-with-custom-path#issuing-api-keys)
if self.caddy_token is not None:
headers["X-Api-Key"] = self.caddy_token
if self.custom_headers:
headers.update(self.custom_headers)
if "http://" in self.base_url:
headers["x-forwarded-proto"] = "https"
headers["x-forwarded-for"] = "127.0.0.1"
return headers
def _prepare_url(self) -> str:
if self.base_url.endswith("/"):
self.base_url = self.base_url[:-1]
if not self.base_url.endswith("/api"):
self.base_url += "/api"
return self.base_url