Source code for stellar_sdk.soroban_server_async

from __future__ import annotations

import json
from typing import TYPE_CHECKING, Type

from . import Keypair
from . import xdr as stellar_xdr
from .account import Account
from .address import Address
from .base_soroban_server import (
    Durability,
    ResourceLeeway,
    _assemble_transaction,
    _generate_unique_request_id,
)
from .client.aiohttp_client import AiohttpClient
from .exceptions import (
    AccountNotFoundException,
    PrepareTransactionException,
    SorobanRpcErrorResponse,
)
from .soroban_rpc import *

if TYPE_CHECKING:
    from .client.base_async_client import BaseAsyncClient
    from .transaction_envelope import TransactionEnvelope

__all__ = ["SorobanServerAsync", "Durability"]

V = TypeVar("V")


[docs] class SorobanServerAsync: """Server handles the network connection to a Soroban RPC instance and exposes an interface for requests to that instance. :param server_url: Soroban RPC server URL. (ex. ``https://soroban-testnet.stellar.org:443``) :param client: A client instance that will be used to make requests. """ def __init__( self, server_url: str = "https://soroban-testnet.stellar.org:443", client: Optional[BaseAsyncClient] = None, ) -> None: self.server_url: str = server_url if not client: client = AiohttpClient() self._client: BaseAsyncClient = client
[docs] async def get_health(self) -> GetHealthResponse: """General node health check. See `Soroban Documentation - getHealth <https://soroban.stellar.org/api/methods/getHealth>`_ :return: A :class:`GetHealthResponse <stellar_sdk.soroban_rpc.GetHealthResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ request: Request = Request( id=_generate_unique_request_id(), method="getHealth", params=None, ) return await self._post(request, GetHealthResponse)
[docs] async def get_events( self, start_ledger: int, filters: Sequence[EventFilter] = None, cursor: str = None, limit: int = None, ) -> GetEventsResponse: """Fetch a list of events that occurred in the ledger range. See `Soroban Documentation - getEvents <https://soroban.stellar.org/api/methods/getEvents>`_ :param start_ledger: The first ledger to include in the results. :param filters: A list of filters to apply to the results. :param cursor: A cursor value for use in pagination. :param limit: The maximum number of records to return. :return: A :class:`GetEventsResponse <stellar_sdk.soroban_rpc.GetEventsResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ pagination = PaginationOptions(cursor=cursor, limit=limit) data = GetEventsRequest( startLedger=str(start_ledger), filters=filters, pagination=pagination, ) request: Request = Request[GetEventsRequest]( id=_generate_unique_request_id(), method="getEvents", params=data ) return await self._post(request, GetEventsResponse)
[docs] async def get_network(self) -> GetNetworkResponse: """General info about the currently configured network. :return: A :class:`GetNetworkResponse <stellar_sdk.soroban_rpc.GetNetworkResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ request: Request = Request( id=_generate_unique_request_id(), method="getNetwork", params=None, ) return await self._post(request, GetNetworkResponse)
[docs] async def get_latest_ledger(self) -> GetLatestLedgerResponse: """Fetches the latest ledger meta info from network which Soroban-RPC is connected to. :return: A :class:`GetLatestLedgerResponse <stellar_sdk.soroban_rpc.GetLatestLedgerResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ request: Request = Request( id=_generate_unique_request_id(), method="getLatestLedger", params=None, ) return await self._post(request, GetLatestLedgerResponse)
[docs] async def get_ledger_entries( self, keys: List[stellar_xdr.LedgerKey] ) -> GetLedgerEntriesResponse: """For reading the current value of ledger entries directly. Allows you to directly inspect the current state of a contract, a contract's code, or any other ledger entry. This is a backup way to access your contract data which may not be available via events or simulateTransaction. See `Soroban Documentation - getLedgerEntries <https://soroban.stellar.org/api/methods/getLedgerEntries>`_ :param keys: The ledger keys to fetch. :return: A :class:`GetLedgerEntriesResponse <stellar_sdk.soroban_rpc.GetLedgerEntryResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ request = Request[GetLedgerEntriesRequest]( id=_generate_unique_request_id(), method="getLedgerEntries", params=GetLedgerEntriesRequest(keys=[key.to_xdr() for key in keys]), ) return await self._post(request, GetLedgerEntriesResponse)
[docs] async def get_transaction(self, transaction_hash: str) -> GetTransactionResponse: """Fetch the specified transaction. See `Soroban Documentation - getTransaction <https://soroban.stellar.org/api/methods/getTransaction>`_ :param transaction_hash: The hash of the transaction to fetch. :return: A :class:`GetTransactionResponse <stellar_sdk.soroban_rpc.GetTransactionResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ request = Request[GetTransactionRequest]( id=_generate_unique_request_id(), method="getTransaction", params=GetTransactionRequest(hash=transaction_hash), ) return await self._post(request, GetTransactionResponse)
[docs] async def simulate_transaction( self, transaction_envelope: TransactionEnvelope, addl_resources: Optional[ResourceLeeway] = None, ) -> SimulateTransactionResponse: """Submit a trial contract invocation to get back return values, expected ledger footprint, and expected costs. See `Soroban Documentation - simulateTransaction <https://soroban.stellar.org/api/methods/simulateTransaction>`_ :param transaction_envelope: The transaction to simulate. It should include exactly one operation, which must be one of :class:`RestoreFootprint <stellar_sdk.operation.RestoreFootprintOperation>`, :class:`InvokeHostFunction <stellar_sdk.operation.InvokeHostFunction>` or :class:`ExtendFootprintTTL <stellar_sdk.operation.RestoreFootprint>` operation. Any provided footprint will be ignored. :param addl_resources: Additional resource include in the simulation. :return: A :class:`SimulateTransactionResponse <stellar_sdk.soroban_rpc.SimulateTransactionResponse>` object contains the cost, footprint, result/auth requirements (if applicable), and error of the transaction. """ xdr = ( transaction_envelope if isinstance(transaction_envelope, str) else transaction_envelope.to_xdr() ) resource_config = None if addl_resources: resource_config = ResourceConfig( instruction_lee_way=addl_resources.cpu_instructions, ) request = Request[SimulateTransactionRequest]( id=_generate_unique_request_id(), method="simulateTransaction", params=SimulateTransactionRequest( transaction=xdr, resource_config=resource_config ), ) return await self._post(request, SimulateTransactionResponse)
[docs] async def send_transaction( self, transaction_envelope: Union[TransactionEnvelope, str] ) -> SendTransactionResponse: """Submit a real transaction to the Stellar network. This is the only way to make changes "on-chain". See `Soroban Documentation - sendTransaction <https://soroban.stellar.org/api/methods/sendTransaction>`_ :param transaction_envelope: The transaction to send. :return: A :class:`SendTransactionResponse <stellar_sdk.soroban_rpc.SendTransactionResponse>` object. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ xdr = ( transaction_envelope if isinstance(transaction_envelope, str) else transaction_envelope.to_xdr() ) request = Request[SendTransactionRequest]( id=_generate_unique_request_id(), method="sendTransaction", params=SendTransactionRequest(transaction=xdr), ) return await self._post(request, SendTransactionResponse)
[docs] async def load_account(self, account_id: str) -> Account: """Load an account from the server, you can use the returned account object as source account for transactions. :param account_id: The account ID. :return: An :class:`Account <stellar_sdk.account.Account>` object. :raises: :exc:`AccountNotFoundException <stellar_sdk.exceptions.AccountNotFoundException>` - If the account is not found on the network. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ account_id_xdr = Keypair.from_public_key(account_id).xdr_account_id() key = stellar_xdr.LedgerKey( stellar_xdr.LedgerEntryType.ACCOUNT, account=stellar_xdr.LedgerKeyAccount(account_id=account_id_xdr), ) resp = await self.get_ledger_entries([key]) if not resp.entries: raise AccountNotFoundException(account_id) assert len(resp.entries) == 1 data = stellar_xdr.LedgerEntryData.from_xdr(resp.entries[0].xdr) assert data.account is not None return Account(account_id, data.account.seq_num.sequence_number.int64)
[docs] async def get_contract_data( self, contract_id: str, key: stellar_xdr.SCVal, durability: Durability = Durability.PERSISTENT, ) -> Optional[LedgerEntryResult]: """Reads the current value of contract data ledger entries directly. :param contract_id: The contract ID containing the data to load. Encoded as Stellar Contract Address, for example: ``"CCJZ5DGASBWQXR5MPFCJXMBI333XE5U3FSJTNQU7RIKE3P5GN2K2WYD5"`` :param key: The key of the contract data to load. :param durability: The "durability keyspace" that this ledger key belongs to, which is either :class:`Durability.TEMPORARY` or :class:`Durability.PERSISTENT`. Defaults to :class:`Durability.PERSISTENT`. :return: A :class:`LedgerEntryResult <stellar_sdk.soroban_rpc.LedgerEntryResult>` object contains the ledger entry result or ``None`` if not found. :raises: :exc:`SorobanRpcErrorResponse <stellar_sdk.exceptions.SorobanRpcErrorResponse>` - If the Soroban-RPC instance returns an error response. """ sc_address = Address(contract_id).to_xdr_sc_address() xdr_durability = ( stellar_xdr.ContractDataDurability.PERSISTENT if durability == Durability.PERSISTENT else stellar_xdr.ContractDataDurability.TEMPORARY ) contract_key = stellar_xdr.LedgerKey( stellar_xdr.LedgerEntryType.CONTRACT_DATA, contract_data=stellar_xdr.LedgerKeyContractData( contract=sc_address, key=key, durability=xdr_durability, ), ) resp = await self.get_ledger_entries([contract_key]) entries = resp.entries if not entries: return None return entries[0]
[docs] async def prepare_transaction( self, transaction_envelope: TransactionEnvelope, simulate_transaction_response: SimulateTransactionResponse = None, ) -> TransactionEnvelope: """Submit a trial contract invocation, first run a simulation of the contract invocation as defined on the incoming transaction, and apply the results to a new copy of the transaction which is then returned. Setting the ledger footprint and authorization, so the resulting transaction is ready for signing and sending. The returned transaction will also have an updated fee that is the sum of fee set on incoming transaction with the contract resource fees estimated from simulation. It is advisable to check the fee on returned transaction and validate or take appropriate measures for interaction with user to confirm it is acceptable. You can call the :meth:`simulate_transaction` method directly first if you want to inspect estimated fees for a given transaction in detail first if that is of importance. :param transaction_envelope: The transaction to prepare. It should include exactly one operation, which must be one of :py:class:`RestoreFootprint <stellar_sdk.operation.RestoreFootprint>`, :py:class:`ExtendFootprintTTL <stellar_sdk.operation.ExtendFootprintTTL>`, or :py:class:`InvokeHostFunction <stellar_sdk.operation.InvokeHostFunction>`. Any provided footprint will be ignored. You can use :meth:`stellar_sdk.Transaction.is_soroban_transaction` to check if a transaction is a Soroban transaction. Any provided footprint will be overwritten. However, if your operation has existing auth entries, they will be preferred over ALL auth entries from the simulation. In other words, if you include auth entries, you don't care about the auth returned from the simulation. Other fields (footprint, etc.) will be filled as normal. :param simulate_transaction_response: The response of the simulation of the transaction, typically you don't need to pass this parameter, it will be automatically called if you don't pass it. :return: A copy of the :class:`TransactionEnvelope <stellar_sdk.transaction_envelope.TransactionEnvelope>`, with the expected authorizations (in the case of invocation) and ledger footprint added. The transaction fee will also automatically be padded with the contract's minimum resource fees discovered from the simulation. """ if not simulate_transaction_response: simulate_transaction_response = await self.simulate_transaction( transaction_envelope ) if simulate_transaction_response.error: raise PrepareTransactionException( "Simulation transaction failed, the response contains error information.", simulate_transaction_response, ) te = _assemble_transaction(transaction_envelope, simulate_transaction_response) return te
async def _post(self, request_body: Request, response_body_type: Type[V]) -> V: json_data = request_body.model_dump_json(by_alias=True) data = await self._client.post( self.server_url, json_data=json.loads(json_data), ) response = Response[response_body_type].model_validate(data.json()) # type: ignore[valid-type] if response.error: raise SorobanRpcErrorResponse( response.error.code, response.error.message, response.error.data ) assert response.result is not None return response.result
[docs] async def close(self) -> None: """Close underlying connector, and release all acquired resources.""" await self._client.close()
async def __aenter__(self) -> "SorobanServerAsync": return self async def __aexit__(self, exc_type, exc_val, exc_tb) -> None: await self.close() def __str__(self): return f"<SorobanServerAsync [server_url={self.server_url}, client={self._client}]>"