Working With Balances
Core concepts
Each contract has a balance
property: both derivatives of gl.Contract
and gl.ContractAt
All transactions and messages (both internal and external) can have attached value
: ContractAt(...).emit(value=100).my_method()
To get value
from the incoming message users can read gl.message.value
There is a special method to call a "nameless" transfer: ContractAt(...).emit_transfer(value=100)
.
You can receive such transfers by overriding __receive__(self)
method.
There are few simple rules:
- If your method expects a
value
, annotate it with@gl.public.write.payable
instead of justwrite
- Emitting transaction to a non-payable method with non-zero value is an error
- If transaction outcome is not success, attached value is sent back
Code example
working_with_balances
class TokenForwarder(gl.Contract):
vault_contract: Address
def __init__(self, vault_contract: str):
self.vault_contract = Address(vault_contract)
@gl.public.write.payable
def forward(self) -> None:
vault = gl.ContractAt(self.vault_contract)
amount = gl.message.value
vault.emit_transfer(value=amount)
@gl.public.write.payable
def donate_twice(self) -> bool:
vault = gl.ContractAt(self.vault_contract)
amount = gl.message.value
if self.balance < amount:
return False
vault.emit(value=amount*2).save(self.address)
return True
Special methods
There are two special methods allowed in each contract definition:
class Contract(gl.Contract):
@gl.public.write # .payable?
def __handle_undefined_method__(
self, method_name: str, args: list[typing.Any], kwargs: dict[str, typing.Any]
):
"""
Method that is called for undefined method calls,
must be either ``@gl.public.write`` or ``@gl.public.write.payable``
"""
...
@gl.public.write.payable
def __receive__(self):
"""
Method that is called for no-method transfers,
must be ``@gl.public.write.payable``
"""
...
Below is a diagram that shows how GenVM decides which method to pick, in case any regular method did not match: