Source code for multiversx_sdk.native_auth.native_auth_client

import base64
import json
from typing import Any, Optional

import requests

from multiversx_sdk.core.address import Address
from multiversx_sdk.native_auth.config import NativeAuthClientConfig
from multiversx_sdk.native_auth.errors import NativeAuthClientError


[docs] class NativeAuthClient: def __init__(self, config: Optional[NativeAuthClientConfig] = None) -> None: self.config = config or NativeAuthClientConfig()
[docs] def initialize(self, extra_info: Optional[dict[Any, Any]] = None) -> str: extra_info = extra_info if extra_info else {} block_hash = self.get_current_block_hash() encoded_extra_info = self._encode_value(json.dumps(extra_info)) encoded_origin = self._encode_value(self.config.origin) return f"{encoded_origin}.{block_hash}.{self.config.expiry_seconds}.{encoded_extra_info}"
[docs] def get_token_for_signing(self, address: Address, init_token: str) -> bytes: return f"{address.to_bech32()}{init_token}".encode()
[docs] def get_token(self, address: Address, token: str, signature: str) -> str: encoded_address = self._encode_value(address.to_bech32()) encoded_token = self._encode_value(token) return f"{encoded_address}.{encoded_token}.{signature}"
[docs] def get_current_block_hash(self) -> str: if self.config.gateway_url: return self._get_current_block_hash_using_gateway() return self._get_current_block_hash_using_api()
def _get_current_block_hash_using_gateway(self) -> str: round = self._get_current_round() url = f"{self.config.gateway_url}/blocks/by-round/{round}" response = self._execute_request(url) blocks: list[dict[str, Any]] = response["data"]["blocks"] block: dict[str, str] = [b for b in blocks if b["shard"] == self.config.block_hash_shard][0] return block["hash"] def _get_current_round(self) -> int: if self.config.gateway_url is None: raise NativeAuthClientError("Gateway URL not set") if self.config.block_hash_shard is None: raise NativeAuthClientError("Blockhash shard not set") url = f"{self.config.gateway_url}/network/status/{self.config.block_hash_shard}" response = self._execute_request(url) status: dict[str, int] = response["data"]["status"] return status["erd_current_round"] def _get_current_block_hash_using_api(self) -> str: try: url = f"{self.config.api_url}/blocks/latest?ttl={self.config.expiry_seconds}&fields=hash" response: dict[str, str] = self._execute_request(url) if response["hash"]: return response["hash"] except Exception: pass return self._get_current_block_hash_using_api_fallback() def _get_current_block_hash_using_api_fallback(self) -> str: url = f"{self.config.api_url}/blocks?size=1&fields=hash" if self.config.block_hash_shard: url += f"&shard={self.config.block_hash_shard}" response: list[dict[str, str]] = self._execute_request(url) return response[0]["hash"] def _encode_value(self, string: str) -> str: encoded = base64.b64encode(string.encode("utf-8")).decode("utf-8") return self._escape(encoded) def _escape(self, string: str) -> str: return string.replace("+", "-").replace("/", "_").replace("=", "") def _execute_request(self, url: str) -> Any: response = requests.get(url=url, headers=self.config.extra_request_headers) response.raise_for_status() return response.json()