Source code for stellar_sdk.soroban_server

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.requests_client import RequestsClient
from .exceptions import (
    AccountNotFoundException,
    PrepareTransactionException,
    SorobanRpcErrorResponse,
)
from .soroban_rpc import *

if TYPE_CHECKING:
    from .client.base_sync_client import BaseSyncClient
    from .transaction_envelope import TransactionEnvelope

__all__ = ["SorobanServer", "Durability"]

V = TypeVar("V")


[docs] class SorobanServer: """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[BaseSyncClient] = None, ) -> None: self.server_url: str = server_url if not client: client = RequestsClient() self._client: BaseSyncClient = client
[docs] 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 self._post(request, GetHealthResponse)
[docs] 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 self._post(request, GetEventsResponse)
[docs] 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 self._post(request, GetNetworkResponse)
[docs] 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 self._post(request, GetLatestLedgerResponse)
[docs] 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 self._post(request, GetLedgerEntriesResponse)
[docs] 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 self._post(request, GetTransactionResponse)
[docs] 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 self._post(request, SimulateTransactionResponse)
[docs] 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 self._post(request, SendTransactionResponse)
[docs] 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 = 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] 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 = self.get_ledger_entries([contract_key]) entries = resp.entries if not entries: return None return entries[0]
[docs] 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 = 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
[docs] def close(self) -> None: """Close underlying connector, and release all acquired resources.""" self._client.close()
def _post(self, request_body: Request, response_body_type: Type[V]) -> V: json_data = request_body.model_dump_json(by_alias=True) data = 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 def __enter__(self) -> "SorobanServer": return self def __exit__(self, exc_type, exc_val, exc_tb): self.close() def __str__(self): return f"<SorobanServer [server_url={self.server_url}, client={self._client}]>"