Skip to content

Commit

Permalink
basic implementation of fungible asset client
Browse files Browse the repository at this point in the history
  • Loading branch information
fishronsage authored and 流戈 committed Jun 18, 2024
1 parent e5e1ecb commit 57e5bd4
Show file tree
Hide file tree
Showing 2 changed files with 414 additions and 1 deletion.
175 changes: 174 additions & 1 deletion aptos_sdk/aptos_token_client.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@

from __future__ import annotations

from typing import Any, List, Tuple
from typing import Any, List, Optional, Tuple

from .account import Account
from .account_address import AccountAddress
Expand Down Expand Up @@ -298,13 +298,83 @@ def parse(resource: dict[str, Any]) -> PropertyMap:
return PropertyMap(properties)


class FAConcurrentSupply:
value: int
max_value: int

struct_tag = "0x1::fungible_asset::ConcurrentSupply"

def __init__(self, value: int, max_value: int) -> None:
self.value = value
self.max_value = max_value

def __str__(self) -> str:
return f"FAConcurrentSupply[value: {self.value}, max_value: {self.max_value}]"

@staticmethod
def parse(resource: dict[str, Any]) -> FAConcurrentSupply:
return FAConcurrentSupply(
int(resource["current"]["value"]), int(resource["current"]["max_value"])
)


class FAMetadata:
name: str
symbol: str
decimals: int
icon_uri: str
project_uri: str

def __init__(
self, name: str, symbol: str, decimals: int, icon_uri: str, project_uri: str
) -> None:
self.name = name
self.symbol = symbol
self.decimals = decimals
self.icon_uri = icon_uri
self.project_uri = project_uri

def __str__(self) -> str:
return f"FAMetadata[name: {self.name}, symbol: {self.symbol}, decimals: {self.decimals}, icon_uri: {self.icon_uri}, project_uri: {self.project_uri}]"

@staticmethod
def parse(resource: dict[str, Any]) -> FAMetadata:
return FAMetadata(
resource["name"],
resource["symbol"],
int(resource["decimals"]),
resource["icon_uri"],
resource["project_uri"],
)


class FungibleStore:
balance: int
frozen: bool

struct_tag = "0x1::fungible_asset::FungibleStore"

def __init__(self, balance: int, frozen: bool) -> None:
self.balance = balance
self.frozen = frozen

def __str__(self):
return f"FungibleStore[balance: {self.balance}, frozen: {self.frozen}]"

@staticmethod
def parse(resource: dict[str, Any]) -> FungibleStore:
return FungibleStore(int(resource["balance"]), resource["frozen"])


class ReadObject:
resource_map: dict[str, Any] = {
Collection.struct_tag: Collection,
Object.struct_tag: Object,
PropertyMap.struct_tag: PropertyMap,
Royalty.struct_tag: Royalty,
Token.struct_tag: Token,
FAConcurrentSupply.struct_tag: FAConcurrentSupply,
FungibleStore.struct_tag: FungibleStore,
}

resources: dict[Any, Any]
Expand Down Expand Up @@ -631,3 +701,106 @@ async def tokens_minted_from_transaction(
continue
mints.append(AccountAddress.from_str_relaxed(event["data"]["token"]))
return mints


class FungibleAssetClient:
"""A wrapper around reading and mutating Fungible Assets"""

def __init__(self, rest_client: RestClient):
self.client = rest_client

async def __primary_store_view(
self,
function: str,
args: List[TransactionArgument],
ledger_version: Optional[int] = None,
) -> Any:
module = "0x1::primary_fungible_store"
ty_args = [TypeTag(StructTag.from_str("0x1::fungible_asset::Metadata"))]
return await self.client.view_bcs_payload(
module, function, ty_args, args, ledger_version
)

async def read_object(self, address: AccountAddress) -> ReadObject:
resources = {}
read_resources = await self.client.account_resources(address)
for resource in read_resources:
if resource["type"] in ReadObject.resource_map:
resource_obj = ReadObject.resource_map[resource["type"]]
resources[resource_obj] = resource_obj.parse(resource["data"])
return ReadObject(resources)

async def transfer(
self,
sender: Account,
metadata_address: AccountAddress,
receiver_address: AccountAddress,
amount: int,
sequence_number: Optional[int] = None,
) -> str:
"""Transfer amount of fungible asset from sender's primary store to receiver's primary store."""
payload = EntryFunction.natural(
"0x1::primary_fungible_store",
"transfer",
[TypeTag(StructTag.from_str("0x1::fungible_asset::Metadata"))],
[
TransactionArgument(metadata_address, Serializer.struct),
TransactionArgument(receiver_address, Serializer.struct),
TransactionArgument(amount, Serializer.u64),
],
)
signed_transaction = await self.client.create_bcs_signed_transaction(
sender, TransactionPayload(payload), sequence_number=sequence_number
)
return await self.client.submit_bcs_transaction(signed_transaction)

async def balance(
self,
metadata_address: AccountAddress,
address: AccountAddress,
ledger_version: Optional[int] = None,
) -> int:
"""Get the balance of account's primary store."""
resp = await self.__primary_store_view(
"balance",
[
TransactionArgument(address, Serializer.struct),
TransactionArgument(metadata_address, Serializer.struct),
],
ledger_version,
)
return int(resp[0])

async def is_frozen(
self,
metadata_address: AccountAddress,
address: AccountAddress,
ledger_version: Optional[int] = None,
) -> bool:
"""Return whether the given account's primary store is frozen."""
resp = await self.__primary_store_view(
"is_frozen",
[
TransactionArgument(address, Serializer.struct),
TransactionArgument(metadata_address, Serializer.struct),
],
ledger_version,
)
return resp[0]

async def primary_store_address(
self,
metadata_address,
address: AccountAddress,
ledger_version: Optional[int] = None,
) -> str:
"""Get the address of the primary store for the given account."""
resp = await self.__primary_store_view(
"primary_store_address",
[
TransactionArgument(address, Serializer.struct),
TransactionArgument(metadata_address, Serializer.struct),
],
ledger_version,
)
return resp[0]
Loading

0 comments on commit 57e5bd4

Please sign in to comment.