Skip to content

Commit

Permalink
feat: protocol v0.1.0 support
Browse files Browse the repository at this point in the history
  • Loading branch information
markuslevonyak committed Sep 20, 2024
1 parent 168a01f commit 82ffbb6
Show file tree
Hide file tree
Showing 20 changed files with 1,968 additions and 1,233 deletions.
15 changes: 13 additions & 2 deletions Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -61,8 +61,19 @@ test:
coverage:
poetry run python3 -m pytest --cov-report term-missing --cov=pantos tests

.PHONY: local-common
local-common:
ifndef DEV_PANTOS_COMMON
$(error Please define DEV_PANTOS_COMMON variable)
endif
$(eval CURRENT_COMMON := $(shell echo .venv/lib/python3.*/site-packages/pantos/common))
@if [ -d "$(CURRENT_COMMON)" ]; then \
rm -rf "$(CURRENT_COMMON)"; \
ln -s "$(DEV_PANTOS_COMMON)" "$(CURRENT_COMMON)"; \
else \
echo "Directory $(CURRENT_COMMON) does not exist"; \
fi

.PHONY: clean
clean:
rm -r -f build/
rm -r -f dist/
rm -r -f pantos_common.egg-info/
3 changes: 3 additions & 0 deletions pantos/client-library.env
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
# protocol #
# PROTOCOL_MAINNET=
# PROTOCOL_TESTNET=
# token_creator #
# TOKEN_CREATOR_URL=
# service_nodes #
Expand Down
4 changes: 4 additions & 0 deletions pantos/client-library.yml
Original file line number Diff line number Diff line change
@@ -1,3 +1,7 @@
protocol:
mainnet: !ENV ${PROTOCOL_MAINNET:1.0.0}
testnet: !ENV ${PROTOCOL_TESTNET:0.1.0}

token_creator:
url: !ENV ${TOKEN_CREATOR_URL:https://creatorb.testnet.pantos.io}

Expand Down
27 changes: 21 additions & 6 deletions pantos/client/library/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,31 +4,46 @@
import ctypes as _ctypes
import multiprocessing as _multiprocessing

import semantic_version as _semantic_version # type: ignore
from pantos.common.configuration import ConfigError as _ConfigError

from pantos.client.library.configuration import config as _config
from pantos.client.library.configuration import load_config as _load_config
from pantos.client.library.exceptions import \
ClientLibraryError as _ClientLibraryError
from pantos.client.library.protocol import \
is_supported_protocol_version as _is_supported_protocol_version

_initialized = _multiprocessing.Value(_ctypes.c_bool, False)


def initialize_library() -> None:
def initialize_library(mainnet: bool) -> None:
"""Initialize the Pantos client library. The function is thread-safe
and performs the initialization only once at the first invocation.
Parameters
----------
mainnet : bool
If True, the client library is initialized for mainnet
operation. Otherwise, it is initialized for testnet operation.
Raises
------
ClientLibraryError
If the library cannot be initialized.
"""
with _initialized.get_lock():
if not _initialized.value: # type: ignore
if not _initialized.value:
try:
_load_config()
except _ConfigError:
raise _ClientLibraryError("error loading config")
_initialized.value = True # type: ignore
# _multiprocessing.Value type bug:
# https://github.com/python/mypy/issues/12299
raise _ClientLibraryError('error loading config')
environment = 'mainnet' if mainnet else 'testnet'
protocol_version = _semantic_version.Version(
_config['protocol'][environment])
if not _is_supported_protocol_version(protocol_version):
raise _ClientLibraryError(
'unsupported Pantos protocol version',
protocol_version=protocol_version)
_initialized.value = True
77 changes: 47 additions & 30 deletions pantos/client/library/api.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@
'get_token_transfer_status', 'deploy_pantos_compatible_token'
]

import typing as _typing
import uuid as _uuid

from pantos.common.blockchains.base import Blockchain
Expand Down Expand Up @@ -69,15 +68,15 @@ def decrypt_private_key(blockchain: Blockchain, keystore: str,
If the private key cannot be loaded from the keystore file.
"""
_initialize_library()
_initialize_library(False)
return _get_blockchain_client(blockchain).decrypt_private_key(
keystore, password)


def retrieve_service_node_bids(
source_blockchain: Blockchain, destination_blockchain: Blockchain,
return_fee_in_main_unit: bool = True) \
-> _typing.Dict[BlockchainAddress, _typing.List[ServiceNodeBid]]:
return_fee_in_main_unit: bool = True, *, mainnet: bool = False) \
-> dict[BlockchainAddress, list[ServiceNodeBid]]:
"""Retrieve the service node bids for token transfers from a
specified source blockchain to a specified destination blockchain.
Expand All @@ -87,10 +86,13 @@ def retrieve_service_node_bids(
The source blockchain of the service node bids.
destination_blockchain : Blockchain
The destination blockchain of the service node bids.
return_fee_in_main_unit : bool
return_fee_in_main_unit : bool, optional
True if the service node bids' fee is to be returned in the
Pantos Token's main unit, False if it is to be returned in the
Pantos Token's smallest subunit (default: True).
mainnet : bool, optional
If True, the function is executed on mainnet. Otherwise, it is
executed on testnet (default: testnet).
Returns
-------
Expand All @@ -104,14 +106,15 @@ def retrieve_service_node_bids(
If the service node bids cannot be retrieved.
"""
_initialize_library()
_initialize_library(mainnet)
return _BidInteractor().retrieve_service_node_bids(
source_blockchain, destination_blockchain, return_fee_in_main_unit)


def retrieve_token_balance(blockchain: Blockchain, account_id: _AccountId,
token_id: _TokenId = _TOKEN_SYMBOL_PAN,
return_in_main_unit: bool = True) -> _Amount:
return_in_main_unit: bool = True, *,
mainnet: bool = False) -> _Amount:
"""Retrieve the token balance of a blockchain account.
Parameters
Expand All @@ -123,10 +126,13 @@ def retrieve_token_balance(blockchain: Blockchain, account_id: _AccountId,
account.
token_id : BlockchainAddress or TokenSymbol
The address or symbol of the token (default: PAN).
return_in_main_unit : bool
return_in_main_unit : bool, optional
True if the token balance is to be returned in the token's main
unit, False if it is to be returned in the token's smallest
subunit (default: True).
mainnet : bool, optional
If True, the function is executed on mainnet. Otherwise, it is
executed on testnet (default: testnet).
Returns
-------
Expand All @@ -141,18 +147,19 @@ def retrieve_token_balance(blockchain: Blockchain, account_id: _AccountId,
If the token balance cannot be retrieved.
"""
_initialize_library()
_initialize_library(mainnet)
request = _TokenInteractor.RetrieveTokenBalanceRequest(
blockchain, token_id, account_id, return_in_main_unit)
return _TokenInteractor().retrieve_token_balance(request)


def transfer_tokens(
source_blockchain: Blockchain, destination_blockchain: Blockchain,
sender_private_key: PrivateKey, recipient_address: BlockchainAddress,
source_token_id: _TokenId, token_amount: _Amount,
service_node_bid: _typing.Optional[_BlockchainAddressBidPair] = None) \
-> ServiceNodeTaskInfo:
def transfer_tokens(source_blockchain: Blockchain,
destination_blockchain: Blockchain,
sender_private_key: PrivateKey,
recipient_address: BlockchainAddress,
source_token_id: _TokenId, token_amount: _Amount,
service_node_bid: _BlockchainAddressBidPair | None = None,
*, mainnet: bool = False) -> ServiceNodeTaskInfo:
"""Transfer tokens from a sender's account on a source blockchain to
a recipient's account on a (possibly different) destination blockchain.
Expand Down Expand Up @@ -180,6 +187,9 @@ def transfer_tokens(
service node's chosen bid. If none is specified,
the registered service node bid with the lowest
fee for the token transfer is automatically chosen.
mainnet : bool, optional
If True, the function is executed on mainnet. Otherwise, it is
executed on testnet (default: testnet).
Returns
-------
Expand All @@ -192,17 +202,18 @@ def transfer_tokens(
If the token transfer cannot be executed.
"""
_initialize_library()
_initialize_library(mainnet)
request = _TransferInteractor.TransferTokensRequest(
source_blockchain, destination_blockchain, sender_private_key,
recipient_address, source_token_id, token_amount, service_node_bid)
return _TransferInteractor().transfer_tokens(request)


def get_token_transfer_status(
source_blockchain: Blockchain, service_node_address: BlockchainAddress,
service_node_task_id: _uuid.UUID,
blocks_to_search: int | None = None) -> TokenTransferStatus:
def get_token_transfer_status(source_blockchain: Blockchain,
service_node_address: BlockchainAddress,
service_node_task_id: _uuid.UUID,
blocks_to_search: int | None = None, *,
mainnet: bool = False) -> TokenTransferStatus:
"""Get the status of a token transfer process.
Parameters
Expand All @@ -217,6 +228,9 @@ def get_token_transfer_status(
blocks_to_search : int or None
The number of blocks to search for the destination transfer.
If None, the search is performed until the genesis block.
mainnet : bool, optional
If True, the function is executed on mainnet. Otherwise, it is
executed on testnet (default: testnet).
Returns
-------
Expand All @@ -229,21 +243,21 @@ def get_token_transfer_status(
If the destination transfer cannot be found.
"""
_initialize_library()
_initialize_library(mainnet)
request = _TransferInteractor.TokenTransferStatusRequest(
source_blockchain, service_node_address, service_node_task_id,
blocks_to_search)
return _TransferInteractor().get_token_transfer_status(request)


def deploy_pantos_compatible_token(
token_name: str, token_symbol: str, token_decimals: int,
token_pausable: bool, token_burnable: bool, token_supply: int,
deployment_blockchains: _typing.List[Blockchain],
payment_blockchain: Blockchain,
payer_private_key: PrivateKey) -> _uuid.UUID:
"""Deploy a pantos compatible blockchain on all the
deployment_blockchains.
def deploy_pantos_compatible_token(token_name: str, token_symbol: str,
token_decimals: int, token_pausable: bool,
token_burnable: bool, token_supply: int,
deployment_blockchains: list[Blockchain],
payment_blockchain: Blockchain,
payer_private_key: PrivateKey, *,
mainnet: bool = False) -> _uuid.UUID:
"""Deploy a Pantos-compatible token on the given blockchains.
Parameters
----------
Expand All @@ -266,6 +280,9 @@ def deploy_pantos_compatible_token(
payer_private_key : PrivateKey
The unencrypted private key of the payer's account on the
payment_blockchain.
mainnet : bool, optional
If True, the function is executed on mainnet. Otherwise, it is
executed on testnet (default: testnet).
Returns
-------
Expand All @@ -278,7 +295,7 @@ def deploy_pantos_compatible_token(
If the deployment process cannot be executed.
"""
_initialize_library()
_initialize_library(mainnet)
request = _TokenDeploymentInteractor.TokenDeploymentRequest(
token_name, token_symbol, token_decimals, token_pausable,
token_burnable, token_supply, deployment_blockchains,
Expand Down
Loading

0 comments on commit 82ffbb6

Please sign in to comment.