import json
from pathlib import Path
from web3 import Web3, AsyncWeb3
from .config import CONTRACT_ADDRESSES
from .rpc.pairs_cache import PairsCache
from .rpc.asset_parameters import AssetParametersRPC
from .rpc.category_parameters import CategoryParametersRPC
from .rpc.blended import BlendedRPC
from .rpc.fee_parameters import FeeParametersRPC
from .rpc.trading_parameters import TradingParametersRPC
from .rpc.snapshot import SnapshotRPC
from .rpc.trade import TradeRPC
from .utils import decoder
from .feed.feed_client import FeedClient
from .signers.base import BaseSigner
from .signers.local_signer import LocalSigner
from .signers.kms_signer import KMSSigner
[docs]
class TraderClient:
"""
This class provides methods to interact with the Avantis smart contracts.
"""
def __init__(
self,
provider_url,
l1_provider_url="https://eth.llamarpc.com",
signer: BaseSigner = None,
):
"""
Constructor for the TraderClient class.
Args:
provider_url: The URL of the Ethereum node provider.
l1_provider_url (optional): The URL of the L1 Ethereum node provider.
signer (optional): The signer to use for signing transactions.
"""
self.web3 = Web3(
Web3.HTTPProvider(provider_url, request_kwargs={"timeout": 60})
)
self.async_web3 = AsyncWeb3(
AsyncWeb3.AsyncHTTPProvider(provider_url, request_kwargs={"timeout": 60})
)
self.l1_web3 = Web3(
Web3.HTTPProvider(l1_provider_url, request_kwargs={"timeout": 60})
)
self.l1_async_web3 = AsyncWeb3(
AsyncWeb3.AsyncHTTPProvider(l1_provider_url, request_kwargs={"timeout": 60})
)
self.contracts = self.load_contracts()
self.chain_id = self.web3.eth.chain_id
self.utils = {
"decoder": lambda *args, **kwargs: decoder(self.web3, *args, **kwargs)
}
self.pairs_cache = PairsCache(self)
self.asset_parameters = AssetParametersRPC(self)
self.category_parameters = CategoryParametersRPC(self)
self.blended = BlendedRPC(self)
self.fee_parameters = FeeParametersRPC(self)
self.trading_parameters = TradingParametersRPC(self)
self.snapshot = SnapshotRPC(self)
self.trade = TradeRPC(self, FeedClient)
self.signer = signer
[docs]
def load_contract(self, name):
"""
Loads the contract ABI and address from the local filesystem.
Args:
name: The name of the contract.
Returns:
A Contract object.
"""
abi_path = Path(__file__).parent / "abis" / f"{name}.sol" / f"{name}.json"
with open(abi_path) as abi_file:
abi = json.load(abi_file)
address = CONTRACT_ADDRESSES[name]
return self.async_web3.eth.contract(address=address, abi=abi["abi"])
[docs]
def load_contracts(self):
"""
Loads all the contracts mentioned in the config from the local filesystem.
Returns:
A dictionary containing the contract names as keys and the Contract objects as values.
"""
return {name: self.load_contract(name) for name in CONTRACT_ADDRESSES.keys()}
[docs]
async def read_contract(self, contract_name, function_name, *args, decode=True):
"""
Calls a read-only function of a contract.
Args:
contract_name: The name of the contract.
function_name: The name of the function.
args: The arguments to the function.
Returns:
The result of the function call.
"""
contract = self.contracts.get(contract_name)
if not contract:
raise ValueError(f"Contract {contract_name} not found")
raw_data = await contract.functions[function_name](*args).call()
if decode:
return self.utils["decoder"](contract, function_name, raw_data)
return raw_data
[docs]
async def write_contract(self, contract_name, function_name, *args, **kwargs):
"""
Calls a write function of a contract.
Args:
contract_name: The name of the contract.
function_name: The name of the function.
args: The arguments to the function.
Returns:
The transaction hash or the transaction object if signer is None.
"""
contract = self.contracts.get(contract_name)
if not contract:
raise ValueError(f"Contract {contract_name} not found")
if self.has_signer() and "from" not in kwargs:
kwargs["from"] = self.get_signer().get_ethereum_address()
if "chainId" not in kwargs:
kwargs["chainId"] = self.chain_id
if "nonce" not in kwargs:
kwargs["nonce"] = await self.get_transaction_count(kwargs["from"])
transaction = await contract.functions[function_name](*args).build_transaction(
kwargs
)
if not self.has_signer():
return transaction
signed_txn = await self.sign_transaction(transaction)
tx_hash = await self.send_and_get_transaction_hash(signed_txn)
return tx_hash
[docs]
def set_signer(self, signer: BaseSigner):
"""
Sets the signer.
"""
self.signer = signer
[docs]
def get_signer(self):
"""
Gets the signer.
"""
return self.signer
[docs]
def remove_signer(self):
"""
Removes the signer.
"""
self.signer = None
[docs]
def has_signer(self):
"""
Checks if the signer is set.
"""
return self.signer is not None
[docs]
def set_local_signer(self, private_key):
"""
Sets the local signer.
"""
self.signer = LocalSigner(private_key, self.async_web3)
[docs]
def set_aws_kms_signer(self, kms_key_id, region_name="us-east-1"):
"""
Sets the AWS KMS signer.
"""
self.signer = KMSSigner(self.async_web3, kms_key_id, region_name)
[docs]
async def sign_transaction(self, transaction):
"""
Signs a transaction.
Args:
transaction: The transaction object.
Returns:
The signed transaction object.
"""
if not self.has_signer():
raise ValueError(
"No signer is set. Please set a signer using `set_signer`."
)
return await self.signer.sign_transaction(transaction)
[docs]
async def send_and_get_transaction_hash(self, signed_txn):
"""
Gets the transaction hash.
Args:
signed_txn: The signed transaction object.
Returns:
The transaction hash.
"""
return await self.async_web3.eth.send_raw_transaction(signed_txn.rawTransaction)
[docs]
async def wait_for_transaction_receipt(self, tx_hash):
"""
Waits for the transaction to be mined.
Args:
tx_hash: The transaction hash.
Returns:
The transaction receipt.
"""
return await self.async_web3.eth.wait_for_transaction_receipt(tx_hash)
[docs]
async def sign_and_get_receipt(self, transaction):
"""
Signs a transaction and waits for it to be mined.
Args:
transaction: The transaction object.
Returns:
The transaction receipt.
"""
gas_estimate = await self.get_gas_estimate(transaction)
transaction["gas"] = gas_estimate
signed_txn = await self.sign_transaction(transaction)
tx_hash = await self.send_and_get_transaction_hash(signed_txn)
return await self.wait_for_transaction_receipt(tx_hash)
[docs]
async def get_transaction_count(self, address=None):
"""
Gets the transaction count.
Args:
address (optional): The address.
Returns:
The transaction count.
"""
if address is None:
address = self.get_signer().get_ethereum_address()
return await self.async_web3.eth.get_transaction_count(address)
[docs]
async def get_gas_price(self):
"""
Gets the gas price.
Returns:
The gas price.
"""
return await self.async_web3.eth.gas_price
[docs]
async def get_chain_id(self):
"""
Gets the chain id.
Returns:
The chain id.
"""
return await self.async_web3.eth.chain_id
[docs]
async def get_balance(self, address=None):
"""
Gets the balance.
Args:
address (optional): The address.
Returns:
The balance.
"""
if address is None:
address = self.get_signer().get_ethereum_address()
return await self.async_web3.eth.get_balance(address)
[docs]
async def get_usdc_balance(self, address=None):
"""
Gets the USDC balance.
Args:
address (optional): The address.
Returns:
The USDC balance.
"""
if address is None:
address = self.get_signer().get_ethereum_address()
balance = await self.read_contract("USDC", "balanceOf", address, decode=False)
return balance / 10**6
[docs]
async def get_usdc_allowance_for_trading(self, address=None):
"""
Gets the USDC allowance for the Trading Storage contract.
Args:
address (optional): The address.
Returns:
The USDC allowance.
"""
if address is None:
address = self.get_signer().get_ethereum_address()
trading_storage_address = self.contracts["TradingStorage"].address
allowance = await self.read_contract(
"USDC", "allowance", address, trading_storage_address, decode=False
)
return allowance / 10**6
[docs]
async def approve_usdc_for_trading(self, amount=100000):
"""
Approves the USDC amount for the Trading Storage contract.
Args:
amount (optional): The amount to approve. Defaults to $100,000.
Returns:
The transaction hash.
"""
trading_storage_address = self.contracts["TradingStorage"].address
return await self.write_contract(
"USDC", "approve", trading_storage_address, int(amount * 10**6)
)
[docs]
async def get_gas_estimate(self, transaction):
"""
Gets the gas estimate.
Args:
transaction: The transaction object.
Returns:
The gas estimate.
"""
return await self.async_web3.eth.estimate_gas(transaction)
[docs]
async def get_transaction_hex(self, transaction):
"""
Gets the transaction hex.
Args:
transaction: The transaction object.
Returns:
The transaction hex.
"""
return transaction.hex()