ARC-4: Application Binary Interface¶
ARC4 defines a set of encodings and behaviors for authoring and interacting with an Algorand Smart Contract. It is not the only way to author a smart contract, but adhering to it will make it easier for other clients and users to interop with your contract.
To author an arc4 contract you should extend the ARC4Contract
base class.
from algopy import ARC4Contract
class HelloWorldContract(ARC4Contract):
...
ARC-32 and ARC-56¶
ARC32 extends the concepts in ARC4 to include an Application Specification which more holistically describes a smart contract and its associated state.
ARC-32/ARC-56 Application Specification files are automatically generated by the compiler for ARC4
contracts as <ContractName>.arc32.json
or <ContractName>.arc56.json
Methods¶
Individual methods on a smart contract should be annotated with an abimethod
decorator. This decorator is used to indicate a method which should be externally callable. The decorator itself includes properties to restrict when the method should be callable, for instance only when the application is being created or only when the OnComplete action is OptIn.
A method that should not be externally available should be annotated with a subroutine
decorator.
Method docstrings will be used when outputting ARC-32 or ARC-56 application specifications, the following docstrings styles are supported ReST, Google, Numpydoc-style and Epydoc.
from algopy import ARC4Contract, subroutine, arc4
class HelloWorldContract(ARC4Contract):
@arc4.abimethod(create=False, allow_actions=["NoOp", "OptIn"], name="external_name")
def hello(self, name: arc4.String) -> arc4.String:
return self.internal_method() + name
@subroutine
def internal_method(self) -> arc4.String:
return arc4.String("Hello, ")
Router¶
Algorand Smart Contracts only have two possible programs that are invoked when making an ApplicationCall Transaction (appl
). The “clear state” program which is called when using an OnComplete action of ClearState
or the “approval” program which is called for all other OnComplete actions.
Routing is required to dispatch calls handled by the approval program to the relevant ABI methods. When extending ARC4Contract
, the routing code is automatically generated for you by the PuyaPy compiler.
Types¶
ARC4 defines a number of data types which can be used in an ARC4 compatible contract and details how these types should be encoded in binary.
Algorand Python exposes these through a number of types which can be imported from the algopy.arc4
module. These types represent binary encoded values following the rules prescribed in the ARC which can mean operations performed directly on these types are not as efficient as ones performed on natively supported types (such as algopy.UInt64
or algopy.Bytes
)
Where supported, the native equivalent of an ARC4 type can be obtained via the .native
property. It is possible to use native types in an ABI method and the router will automatically encode and decode these types to their ARC4 equivalent.
Booleans¶
Type: algopy.arc4.Bool
Encoding: A single byte where the most significant bit is 1
for True
and 0
for False
Native equivalent: builtins.bool
Unsigned ints¶
Types: algopy.arc4.UIntN
(<= 64 bits) algopy.arc4.BigUIntN
(> 64 bits)
Encoding: A big endian byte array of N bits
Native equivalent: algopy.UInt64
or puya.py.BigUInt
Common bit sizes have also been aliased under algopy.arc4.UInt8
, algopy.arc4.UInt16
etc. A uint of any size between 8 and 512 bits (in intervals of 8bits) can be created using a generic parameter. It can be helpful to define your own alias for this type.
import typing as t
from algopy import arc4
UInt40: t.TypeAlias = arc4.UIntN[t.Literal[40]]
Unsigned fixed point decimals¶
Types: algopy.arc4.UFixedNxM
(<= 64 bits) algopy.arc4.BigUFixedNxM
(> 64 bits)
Encoding: A big endian byte array of N bits where encoded_value = value / (10^M)
Native equivalent: none
import typing as t
from algopy import arc4
Decimal: t.TypeAlias = arc4.UFixedNxM[t.Literal[64], t.Literal[10]]
Bytes and strings¶
Types: algopy.arc4.DynamicBytes
and algopy.arc4.String
Encoding: A variable length byte array prefixed with a 16-bit big endian header indicating the length of the data
Native equivalent: algopy.Bytes
and algopy.String
Strings are assumed to be utf-8 encoded and the length of a string is the total number of bytes, not the total number of characters.
Static arrays¶
Type: algopy.arc4.StaticArray
Encoding: See ARC4 Container Packing
Native equivalent: none
An ARC4 StaticArray is an array of a fixed size. The item type is specified by the first generic parameter and the size is specified by the second.
import typing as t
from algopy import arc4
FourBytes: t.TypeAlias = arc4.StaticArray[arc4.Byte, t.Literal[4]]
Address¶
Type: algopy.arc4.Address
Encoding: A byte array 32 bytes long
Native equivalent: algopy.Account
Address represents an Algorand address’s public key, and can be used instead of algopy.Account
when needing to
reference an address in an ARC4 struct, tuple or return type. It is a subclass of arc4.StaticArray[arc4.Byte, typing.Literal[32]]
Dynamic arrays¶
Type: algopy.arc4.DynamicArray
Encoding: See ARC4 Container Packing
Native equivalent: none
An ARC4 DynamicArray is an array of a variable size. The item type is specified by the first generic parameter. Items can be added and removed via .pop
, .append
, and .extend
.
The current length of the array is encoded in a 16-bit prefix similar to the arc4.DynamicBytes
and arc4.String
types
import typing as t
from algopy import arc4
UInt64Array: t.TypeAlias = arc4.DynamicArray[arc4.UInt64]
Tuples¶
Type: algopy.arc4.Tuple
Encoding: See ARC4 Container Packing
Native equivalent: builtins.tuple
ARC4 Tuples are immutable statically sized arrays of mixed item types. Item types can be specified via generic parameters or inferred from constructor parameters.
Structs¶
Type: algopy.arc4.Struct
Encoding: See ARC4 Container Packing
Native equivalent: typing.NamedTuple
ARC4 Structs are named tuples. The class keyword frozen
can be used to indicate if a struct can be mutated.
Items can be accessed and mutated via names instead of indexes. Structs do not have a .native
property,
but a NamedTuple can be used in ABI methods are will be encoded/decode to an ARC4 struct automatically.
import typing
from algopy import arc4
Decimal: typing.TypeAlias = arc4.UFixedNxM[typing.Literal[64], typing.Literal[9]]
class Vector(arc4.Struct, kw_only=True, frozen=True):
x: Decimal
y: Decimal
ARC4 Container Packing¶
ARC4 encoding rules are detailed explicitly in the ARC. A summary is included here.
Containers are composed of a head and tail portion.
For dynamic arrays, the head is prefixed with the length of the array encoded as a 16-bit number. This prefix is not included in offset calculation
For fixed sized items (eg. Bool, UIntN, or a StaticArray of UIntN), the item is included in the head
Consecutive Bool items are compressed into the minimum number of whole bytes possible by using a single bit to represent each Bool
For variable sized items (eg. DynamicArray, String etc), a pointer is included to the head and the data is added to the tail. This pointer represents the offset from the start of the head to the start of the item data in the tail.
Reference types¶
Types: algopy.Account
, algopy.Application
, algopy.Asset
, algopy.gtxn.PaymentTransaction
, algopy.gtxn.KeyRegistrationTransaction
, algopy.gtxn.AssetConfigTransaction
, algopy.gtxn.AssetTransferTransaction
, algopy.gtxn.AssetFreezeTransaction
, algopy.gtxn.ApplicationCallTransaction
The ARC4 specification allows for using a number of reference types in an ABI method signature where this reference type refers to…
another transaction in the group
an account in the accounts array (
apat
property of the transaction)an asset in the foreign assets array (
apas
property of the transaction)an application in the foreign apps array (
apfa
property of the transaction)
These types can only be used as parameters, and not as return types.
from algopy import (
Account,
Application,
ARC4Contract,
Asset,
arc4,
gtxn,
)
class Reference(ARC4Contract):
@arc4.abimethod
def with_transactions(
self,
asset: Asset,
pay: gtxn.PaymentTransaction,
account: Account,
app: Application,
axfr: gtxn.AssetTransferTransaction
) -> None:
...
Mutability¶
To ensure semantic compatability the compiler will also check for any usages of mutable ARC4 types (arrays and structs) and ensure that any additional references are copied using the .copy()
method.
Python values are passed by reference, and when an object (eg. an array or struct) is mutated in one place, all references to that object see the mutated version. In Python this is managed via the heap.
In Algorand Python these mutable values are instead stored on the stack, so when an additional reference is made (i.e. by assigning to another variable) a copy is added to the stack.
Which means if one reference is mutated, the other references would not see the change.
In order to keep the semantics the same, the compiler forces the addition of .copy()
each time a new reference to the same object to match what will happen on the AVM.
Struct types can be indicated as frozen
which will eliminate the need for a .copy()
as long as the struct also contains no mutable fields (such as arrays or another mutable struct)