ARC69 Minter and Modifier Part 3
Video Tutorial
Contract - User Sends Monster To Training (Stakes) and Removes from Training (Unstakes)
Compile the contract using:
algokit compile arc69NFTmodifier.py
from algopy import ARC4Contract, itxn, Global, GlobalState, UInt64, gtxn, Bytes, subroutine, String, BoxRef, Txn, op, LocalState, Account
from algopy.arc4 import abimethod, Struct, Address, Bool
from algopy.arc4 import UInt64 as arc4UInt64
from algopy.arc4 import String as arc4String
class availableMonster(Struct):
monster_name: arc4String
monster_type: arc4String
monster_description: arc4String
monster_ipfs_hash: arc4String
class userMonsterInfo(Struct):
uid: arc4UInt64
owner: Address
asset_id: arc4UInt64
monster_id: arc4UInt64
level: arc4UInt64
exp: arc4UInt64
training: Bool
training_start_time: arc4UInt64
class arc69NFTmodifier(ARC4Contract):
def __init__(self) -> None:
self.monsterUnitCounter = GlobalState(arc4UInt64(0))
self.monsterCreated = GlobalState(arc4UInt64(0))
@subroutine
def itoa(self, i: UInt64) -> Bytes:
digits = Bytes(b"0123456789")
radix = digits.length
if i < radix:
return digits[i]
return self.itoa(i // radix) + digits[i % radix]
@abimethod
def registerNewMonsterData(
self,
monster_name: arc4String,
monster_type: arc4String,
monster_description: arc4String,
monster_ipfs_hash: arc4String,
payment_txn: gtxn.PaymentTransaction,
) -> arc4String:
# assert payment_txn.amount
assert payment_txn.receiver == Global.current_application_address
self.monsterUnitCounter.value = arc4UInt64(self.monsterUnitCounter.value.native + 1)
box_ref = BoxRef(key=self.monsterUnitCounter.value.bytes)
value, exists = box_ref.maybe()
assert not exists
new_monster_info = availableMonster(
monster_name=monster_name,
monster_type=monster_type,
monster_description=monster_description,
monster_ipfs_hash=monster_ipfs_hash
)
box_ref.create(size=new_monster_info.bytes.length)
box_ref.put(new_monster_info.bytes)
return arc4String('Monster Registered to Contract: ') + monster_name
@abimethod
def determineUserMonster(
self,
payment_txn: gtxn.PaymentTransaction
) -> tuple[arc4String, arc4UInt64]:
assert payment_txn.amount == 18_500
assert payment_txn.receiver == Global.current_application_address
randomizer = arc4UInt64(
Global.latest_timestamp + Global.round + arc4UInt64.from_bytes(Txn.tx_id[0:8]).native % arc4UInt64.from_bytes(Txn.tx_id[-8:]).native
)
hashed_randomizer = op.sha256(randomizer.bytes)
monster_selection = arc4UInt64((arc4UInt64.from_bytes(hashed_randomizer[0:8]).native % self.monsterUnitCounter.value.native) + 1)
users_claim_box = BoxRef(key=Txn.sender.bytes)
value, exists = users_claim_box.maybe()
assert not exists
users_claim_box.create(size=8)
users_claim_box.put(monster_selection.bytes)
return arc4String('Users Monster # Selected:'), monster_selection
@abimethod()
def mintUserMonster(
self,
payment_txn: gtxn.PaymentTransaction
) -> tuple[arc4String, UInt64]:
assert payment_txn.amount >= 152900
assert payment_txn.receiver == Global.current_application_address
users_claim_box = BoxRef(key=Txn.sender.bytes)
claim_box_value, exists = users_claim_box.maybe()
assert exists
monster_selection = arc4UInt64.from_bytes(claim_box_value)
monster_info_bytes = BoxRef(key=monster_selection.bytes)
monster_info_value, exists = monster_info_bytes.maybe()
assert exists
available_monster_info = availableMonster.from_bytes(monster_info_value)
unit_name_int_as_string = String.from_bytes(self.itoa(monster_selection.native))
uid_as_string = String.from_bytes(self.itoa(self.monsterCreated.value.native))
monster_creation_tx = itxn.AssetConfig(
total=1,
manager=Global.current_application_address,
asset_name=available_monster_info.monster_name.native,
unit_name='PK#' + unit_name_int_as_string,
decimals=0,
url= 'https://gateway.pinata.cloud/ipfs/' + available_monster_info.monster_ipfs_hash.native,
note = (
'{"standard": "arc69", "mime_type": "image/png", "properties": {'
'"Type": "' + available_monster_info.monster_type.native + '", '
'"Description": "' + available_monster_info.monster_description.native + '", '
'"Level": "1", '
'"Experience": "0", '
'"Monster Number": "' + unit_name_int_as_string + '", '
'"UID": "' + uid_as_string + '" }}'
),
fee=Global.min_txn_fee
).submit()
user_monster_box = BoxRef(key=self.monsterCreated.value.bytes + Txn.sender.bytes)
user_monster_box_value, exists = user_monster_box.maybe()
assert not exists
new_user_monster_info = userMonsterInfo(
uid=self.monsterCreated.value,
owner=Address(Txn.sender),
asset_id=arc4UInt64(monster_creation_tx.created_asset.id),
monster_id=monster_selection,
level=arc4UInt64(1),
exp=arc4UInt64(0),
training=Bool(False),
training_start_time=arc4UInt64(0),
)
user_monster_box.create(size=new_user_monster_info.bytes.length)
user_monster_box.put(new_user_monster_info.bytes)
users_claim_box.delete()
self.monsterCreated.value = arc4UInt64(self.monsterCreated.value.native + 1)
return arc4String('User monster generated, asset ID: '), monster_creation_tx.created_asset.id
@abimethod
def claimMonster(
self,
opt_in_txn: gtxn.AssetTransferTransaction,
uid: arc4UInt64
) -> String:
user_monster_box = BoxRef(key=uid.bytes + Txn.sender.bytes)
user_monster_box_value, exists = user_monster_box.maybe()
assert exists
users_monster_info = userMonsterInfo.from_bytes(user_monster_box_value)
#assert users_monster_info.owner == Address(Txn.sender)
#assert opt_in_txn.xfer_asset.id == users_monster_info.asset_id
itxn.AssetTransfer(
asset_receiver=Txn.sender,
xfer_asset=users_monster_info.asset_id.native,
fee=Global.min_txn_fee,
asset_amount=1
).submit()
return String('User received their monster')
@abimethod
def trainMonster(
self,
stake_monster_tx: gtxn.AssetTransferTransaction,
uid: arc4UInt64
) -> String:
assert stake_monster_tx.asset_amount == 1
assert stake_monster_tx.asset_receiver == Global.current_application_address
monster_to_train = stake_monster_tx.xfer_asset.id
users_monster_info = BoxRef(key=uid.bytes + Txn.sender.bytes)
value, exists = users_monster_info.maybe()
assert exists
current_monster_info = userMonsterInfo.from_bytes(value)
assert monster_to_train == current_monster_info.asset_id
current_monster_info.training_start_time = arc4UInt64(Global.latest_timestamp)
current_monster_info.training = Bool(True)
users_monster_info.put(current_monster_info.bytes)
return String("Monster Staked")
@abimethod
def removeMonsterFromTraining(
self,
monster_requested_uid: arc4UInt64
) -> String:
users_monster_info_box = BoxRef(key=monster_requested_uid.bytes + Txn.sender.bytes)
users_monster_info_value, exists = users_monster_info_box.maybe()
assert exists
current_users_monster_info = userMonsterInfo.from_bytes(users_monster_info_value)
general_monster_info_box = BoxRef(key=current_users_monster_info.monster_id.bytes)
general_monster_info_value, exists = general_monster_info_box.maybe()
general_monster_info = availableMonster.from_bytes(general_monster_info_value)
unit_name_as_string = String.from_bytes(self.itoa(current_users_monster_info.monster_id.native))
initial_training_time = current_users_monster_info.training_start_time.native
current_time = Global.latest_timestamp
time_difference = current_time - initial_training_time
current_level = current_users_monster_info.level.native
current_experience = current_users_monster_info.exp.native
base_experience = UInt64(200)
cumulative_experience = current_level * base_experience
total_experience = current_experience + time_difference
if total_experience >= cumulative_experience:
levels_gained = (total_experience + base_experience - cumulative_experience) // base_experience
new_level = current_level + levels_gained
new_experience = total_experience - ((new_level - 1) * base_experience)
else:
new_level = current_level
new_experience = total_experience
new_level_string = String.from_bytes(self.itoa(new_level))
new_experience_string = String.from_bytes(self.itoa(new_experience))
uid_as_string = String.from_bytes(self.itoa(current_users_monster_info.uid.native))
itxn.AssetConfig(
sender=Global.current_application_address,
manager=Global.current_application_address,
config_asset=current_users_monster_info.asset_id.native,
note = (
'{"standard": "arc69", "mime_type": "image/png", "properties": {'
'"Type": "' + general_monster_info.monster_type.native + '", '
'"Description": "' + general_monster_info.monster_description.native + '", '
'"Level": "' + new_level_string + '", '
'"Experience": "' + new_experience_string + '", '
'"Monster Number": "' + unit_name_as_string + '", '
'"UID": "' + uid_as_string + '" }}'
),
fee=Global.min_txn_fee
).submit()
itxn.AssetTransfer(
asset_receiver=Txn.sender,
asset_amount=1,
xfer_asset=current_users_monster_info.asset_id.native,
fee=Global.min_txn_fee
).submit()
new_users_monster_info = current_users_monster_info.copy()
new_users_monster_info.level = arc4UInt64(new_level)
new_users_monster_info.exp = arc4UInt64(new_experience)
new_users_monster_info.training = Bool(False)
users_monster_info_box.put(new_users_monster_info.bytes)
return String('Monster Removed from Stake')
Seeing the Current Status of the User's Monster
Note: The UID in the metadata and the user's address is the box name
from algosdk.v2client.algod import AlgodClient
import os
from base64 import b64decode
from algosdk.abi import ABIType
from dotenv import load_dotenv
load_dotenv()
algod_token = os.getenv('algod_token')
algod_server = os.getenv('algod_server')
algod_client = AlgodClient(algod_token, algod_server)
app_id = int(os.getenv('app_id'))
app_boxes = algod_client.application_boxes(app_id)['boxes']
monster_info_coder = ABIType.from_string('(uint64,address,uint64,uint64,uint64,uint64,bool,uint64)')
for box in app_boxes:
box_name = b64decode(box['name'])
if len(box_name) == 40: #This is user specific monster information (box name is uid for monster [8 bytes] + user address [32 bytes])
box_value = algod_client.application_box_by_name(app_id, )['value']
decoded_box_value = monster_info_coder.decode(b64decode(box_value))
print(decoded_box_value)
Executing in Python
The Monster Level-up System
Note: Experience and level gain are calculated based on time staked, with each second providing an additional experience point.
initial_training_time = current_users_monster_info.training_start_time.native
current_time = Global.latest_timestamp
time_difference = current_time - initial_training_time
current_level = current_users_monster_info.level.native
current_experience = current_users_monster_info.exp.native
base_experience = UInt64(200)
cumulative_experience = current_level * base_experience
total_experience = current_experience + time_difference
if total_experience >= cumulative_experience:
levels_gained = (total_experience + base_experience - cumulative_experience) // base_experience
new_level = current_level + levels_gained
new_experience = total_experience - ((new_level - 1) * base_experience)
else:
new_level = current_level
new_experience = total_experience
Staking the Monster
from algosdk.v2client.algod import AlgodClient
from algokit_utils import ApplicationClient
from algosdk.atomic_transaction_composer import AtomicTransactionComposer, AccountTransactionSigner, TransactionWithSigner
from algosdk.transaction import PaymentTxn, AssetTransferTxn
from algosdk.account import address_from_private_key
from pathlib import Path
from algosdk.abi import ABIType
from algosdk.encoding import decode_address
from dotenv import load_dotenv
from base64 import b64decode
import os
load_dotenv()
node_token = os.getenv('algod_token')
node_server = os.getenv('algod_server')
private_key = os.getenv('private_key')
algod_client = AlgodClient(node_token, node_server)
app_spec = Path(__file__).parent / './arc69NFTmodifier.arc32.json'
app_id = int(os.getenv('app_id'))
signer = AccountTransactionSigner(private_key=private_key)
address = address_from_private_key(private_key=private_key)
params = algod_client.suggested_params()
application_client = ApplicationClient(
algod_client=algod_client,
app_spec=app_spec,
app_id=app_id,
signer=signer,
sender=address,
suggested_params=params
)
atc = AtomicTransactionComposer()
decoded_address = decode_address(address)
users_monster = 728462064
stake_monster_tx = AssetTransferTxn(
sender=address,
receiver=application_client.app_address,
sp=params,
amt=1,
index=users_monster
)
wrapped_stake_tx = TransactionWithSigner(stake_monster_tx, signer)
current_uid = (0).to_bytes(8, 'big')
box_ref_1 = current_uid + decoded_address
application_client.compose_call(
atc,
call_abi_method='trainMonster',
stake_monster_tx=wrapped_stake_tx,
uid=0,
transaction_parameters={
'boxes':[[app_id, box_ref_1]],
'foreign_assets': [users_monster]
}
)
results = atc.execute(algod_client, 2)
tx_ids = [results.tx_ids[i] for i in range(len(results.tx_ids))]
abi_results = [results.abi_results[i].return_value for i in range(len(results.abi_results))]
print(tx_ids)
print(abi_results)
Unstaking the Monster
from algosdk.v2client.algod import AlgodClient
from algokit_utils import ApplicationClient
from algosdk.atomic_transaction_composer import AtomicTransactionComposer, AccountTransactionSigner, TransactionWithSigner
from algosdk.transaction import PaymentTxn, AssetTransferTxn
from algosdk.account import address_from_private_key
from pathlib import Path
from algosdk.abi import ABIType
from algosdk.encoding import decode_address
from dotenv import load_dotenv
from base64 import b64decode
import os
load_dotenv()
node_token = os.getenv('algod_token')
node_server = os.getenv('algod_server')
private_key = os.getenv('private_key')
algod_client = AlgodClient(node_token, node_server)
app_spec = Path(__file__).parent / './arc69NFTmodifier.arc32.json'
app_id = int(os.getenv('app_id'))
signer = AccountTransactionSigner(private_key=private_key)
address = address_from_private_key(private_key=private_key)
params = algod_client.suggested_params()
application_client = ApplicationClient(
algod_client=algod_client,
app_spec=app_spec,
app_id=app_id,
signer=signer,
sender=address,
suggested_params=params
)
atc = AtomicTransactionComposer()
decoded_address = decode_address(address)
mbr_fee_payment_tx_1 = PaymentTxn(
sender=address,
sp=params,
receiver=application_client.app_address,
amt=3000,
note='#1'
)
wrapped_payment_tx_1 = TransactionWithSigner(mbr_fee_payment_tx_1, signer)
atc.add_transaction(wrapped_payment_tx_1)
users_monster = 728462064
stake_monster_tx = AssetTransferTxn(
sender=address,
receiver=application_client.app_address,
sp=params,
amt=1,
index=users_monster
)
wrapped_stake_tx = TransactionWithSigner(stake_monster_tx, signer)
current_uid = (0).to_bytes(8, 'big')
box_ref_1 = current_uid + decoded_address
box_ref_2 = (3).to_bytes(8, 'big')
application_client.compose_call(
atc,
call_abi_method='removeMonsterFromTraining',
monster_requested_uid=0,
transaction_parameters={
'boxes':[[app_id, box_ref_1], [app_id, box_ref_2]],
'foreign_assets': [users_monster]
}
)
results = atc.execute(algod_client, 2)
tx_ids = [results.tx_ids[i] for i in range(len(results.tx_ids))]
abi_results = [results.abi_results[i].return_value for i in range(len(results.abi_results))]
print(tx_ids)
print(abi_results)