-
Notifications
You must be signed in to change notification settings - Fork 118
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
ARC-62 - ASA Circulating Supply #302
Merged
Merged
Changes from all commits
Commits
Show all changes
32 commits
Select commit
Hold shift + click to select a range
e29bdfc
doc: add arc62
cusma 240b6c1
doc: markdown lint
cusma 4b5ac18
doc: add reference implementation link
cusma 1bae5b0
feat: add reference implementation assets
cusma 1b7b526
doc: please the ARC linter
cusma a7ace3c
feat: handle closed accounts edge case
cusma 07514a7
feat: specify ARC number in assets
cusma fd31510
test: asset already set
cusma ef705d1
feat: remove unused std errors
cusma eb3803e
doc: normative backwards compatibility
cusma 4227533
doc: add security considerations
cusma d47e298
doc: nit
cusma 6e748fa
doc: nit
cusma 7585d2b
doc: add readonly key
cusma d616f38
doc: add ref. implementation details
cusma 45d0b66
doc: promote to review
cusma ed031b8
doc: generalise labels, add JSON state schema
cusma 539d9f7
feat: generic labels in ref. implementation
cusma e325946
feat: generic labels in ref. implementation
cusma 48a994b
fear: update assets
cusma 0f510cb
feat: remove algokit generator templates
cusma 8a9a468
Status -> Last Call
SudoWeezy b95c6fa
doc: add specs for C2C calls, update "requires" field, conform to new…
cusma 4cffb65
doc: not such a thing as "optional" required ARCs
cusma 21e5b0a
doc: nit
cusma dd80d72
doc: disable direct link to ARC-56 (to be released)
cusma 03afdd4
doc: re-enable direct link to ARC-56
cusma 0265353
doc: relax ARC-56 requirement
cusma c1fc5aa
chore: update assets
cusma a91e7a3
chore: remove ARC-56 dependency
cusma 94bf862
Adding subcategory
SudoWeezy 3f358e6
update last call date
SudoWeezy File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,340 @@ | ||
--- | ||
arc: 62 | ||
title: ASA Circulating Supply | ||
description: Getter method for ASA circulating supply | ||
author: Cosimo Bassi (@cusma) | ||
discussions-to: https://github.com/algorandfoundation/ARCs/issues/302 | ||
status: Last Call | ||
last-call-deadline: 2024-10-15 | ||
type: Standards Track | ||
category: Interface | ||
sub-category: Explorer | ||
created: 2024-06-12 | ||
requires: 2, 4, 22 | ||
--- | ||
|
||
## Abstract | ||
|
||
This ARC introduces a standard for the definition of circulating supply for Algorand | ||
Standard Assets (ASA) and its client-side retrieval. A reference implementation is | ||
suggested. | ||
|
||
## Motivation | ||
|
||
Algorand Standard Asset (ASA) `total` supply is _defined_ upon ASA creation. | ||
|
||
Creating an ASA on the ledger _does not_ imply its `total` supply is immediately | ||
“minted” or “circulating”. In fact, the semantic of token “minting” on Algorand is | ||
slightly different from other blockchains: it is not coincident with the token units | ||
creation on the ledger. | ||
|
||
The Reserve Address, one of the 4 addresses of ASA Role-Based-Access-Control (RBAC), | ||
is conventionally used to identify the portion of `total` supply not yet in circulation. | ||
The Reserve Address has no “privilege” over the token: it is just a “logical” label | ||
used (client-side) to classify an existing amount of ASA as “not in circulation”. | ||
|
||
According to this convention, “minting” an amount of ASA units is equivalent to | ||
_moving that amount out of the Reserve Address_. | ||
|
||
> ASA may have the Reserve Address assigned to a Smart Contract to enforce specific | ||
> “minting” policies, if needed. | ||
|
||
This convention led to a simple and unsophisticated semantic of ASA circulating | ||
supply, widely adopted by clients (wallets, explorers, etc.) to provide standard | ||
information: | ||
|
||
```text | ||
circulating_supply = total - reserve_balance | ||
``` | ||
|
||
Where `reserve_balance` is the ASA balance hold by the Reserve Address. | ||
|
||
However, the simplicity of such convention, who fostered adoption across the Algorand | ||
ecosystem, poses some limitations. Complex and sophisticated use-cases of ASA, such | ||
as regulated stable-coins and tokenized securities among others, require more | ||
detailed and expressive definitions of circulating supply. | ||
|
||
As an example, an ASA could have “burned”, “locked” or “pre-minted” amounts of token, | ||
not held in the Reserve Address, which _should not_ be considered as “circulating” | ||
supply. This is not possible with the basic ASA protocol convention. | ||
|
||
This ARC proposes a standard ABI _read-only_ method (getter) to provide the circulating | ||
supply of an ASA. | ||
|
||
## Specification | ||
|
||
The keywords "**MUST**", "**MUST NOT**", "**REQUIRED**", "**SHALL**", "**SHALL NOT**", | ||
"**SHOULD**", "**SHOULD NOT**", "**RECOMMENDED**", "**MAY**", and "**OPTIONAL**" | ||
in this document are to be interpreted as described in <a href="https://datatracker.ietf.org/doc/html/rfc2119">RFC 2119</a>. | ||
|
||
> Notes like this are non-normative. | ||
|
||
### ABI Method | ||
|
||
A compliant ASA, whose circulating supply definition conforms to this ARC, **MUST** | ||
implement the following method on an Application (referred as _Circulating Supply | ||
App_ in this specification): | ||
|
||
```json | ||
{ | ||
"name": "arc62_get_circulating_supply", | ||
"readonly": true, | ||
"args": [ | ||
{ | ||
"type": "uint64", | ||
"name": "asset_id", | ||
"desc": "ASA ID of the circulating supply" | ||
} | ||
], | ||
"returns": { | ||
"type": "uint64", | ||
"desc": "ASA circulating supply" | ||
}, | ||
"desc": "Get ASA circulating supply" | ||
} | ||
``` | ||
|
||
The `arc62_get_circulating_supply` **MUST** be a _read-only_ ([ARC-22](./arc-0022.md)) | ||
method (getter). | ||
|
||
### Usage | ||
|
||
Getter calls **SHOULD** be _simulated_. | ||
|
||
Any external resources used by the implementation **SHOULD** be discovered and | ||
auto-populated by the simulated getter call. | ||
|
||
#### Example 1 | ||
|
||
> Let the ASA have `total` supply and a Reserve Address (i.e. not set to `ZeroAddress`). | ||
> | ||
> Let the Reserve Address be assigned to an account different from the Circulating | ||
> Supply App Account. | ||
> | ||
> Let `burned` be an external Burned Address dedicated to ASA burned supply. | ||
> | ||
> Let `locked` be an external Locked Address dedicated to ASA locked supply. | ||
> | ||
> The ASA issuer defines the _circulating supply_ as: | ||
> | ||
> ```text | ||
> circulating_supply = total - reserve_balance - burned_balance - locked_balance | ||
> ``` | ||
> | ||
> In this case the simulated read-only method call would auto-populate 1 external | ||
> reference for the ASA and 3 external reference accounts (Reserve, Burned and Locked). | ||
|
||
#### Example 2 | ||
|
||
> Let the ASA have `total` supply and _no_ Reserve Address (i.e. set to `ZeroAddress`). | ||
> | ||
> Let `non_circulating_amount` be a UInt64 Global Var defined by the implementation | ||
> of the Circulating Supply App. | ||
> | ||
> The ASA issuer defines the _circulating supply_ as: | ||
> | ||
> ```text | ||
> circulating_supply = total - non_circulating_amount | ||
> ``` | ||
> | ||
> In this case the simulated read-only method call would auto-populate just 1 external | ||
> reference for the ASA. | ||
|
||
### Circulating Supply Application discovery | ||
|
||
> Given an ASA ID, clients (wallet, explorer, etc.) need to discover the related | ||
> Circulating Supply App. | ||
|
||
An ASA conforming to this ARC **MUST** specify the Circulating Supply App ID. | ||
|
||
> To avoid ecosystem fragmentation this ARC does not propose any new method to specify | ||
> the metadata of an ASA. Instead, it only extends already existing standards. | ||
|
||
If the ASA also conforms to any ARC that supports additional `properties` ([ARC-3](./arc-0003.md), | ||
[ARC-19](./arc-0019.md), [ARC-69](./arc-0069.md)), | ||
then it **MUST** include a `arc-62` key and set the corresponding value to a map, | ||
including the ID of the Circulating Supply App as a value for the key `application-id`. | ||
|
||
#### Example | ||
|
||
```json | ||
{ | ||
//... | ||
"properties": { | ||
//... | ||
"arc-62": { | ||
"application-id": 123 | ||
} | ||
} | ||
//... | ||
} | ||
``` | ||
|
||
## Rationale | ||
|
||
The definition of _circulating supply_ for sophisticated use-cases is usually ASA-specific. | ||
It could involve, for example, complex math or external accounts’ balances, variables | ||
stored in boxes or in global state, etc.. | ||
|
||
For this reason, the proposed method’s signature does not require any reference | ||
to external resources, a part form the `asset_id` of the ASA for which the circulating | ||
supply is defined. | ||
|
||
Eventual external resources can be discovered and auto-populated directly by the | ||
simulated method call. | ||
|
||
The rational of this design choice is avoiding fragmentation and integration overhead | ||
for clients (wallets, explorers, etc.). | ||
|
||
Clients just need to know: | ||
|
||
1. The ASA ID; | ||
1. The Circulating Supply App ID implementing the `arc62_get_circulating_supply` | ||
method for that ASA. | ||
|
||
## Backwards Compatibility | ||
|
||
Existing ASA willing to conform to this ARC **MUST** specify the Circulating Supply | ||
App ID as [ARC-2](./arc-0002.md) `AssetConfig` transaction note field, as follows: | ||
|
||
- The `<arc-number>` **MUST** be equal to `62`; | ||
- The **RECOMMENDED** `<data-format>` are <a href="https://msgpack.org/">MsgPack</a> | ||
(`m`) or <a href="https://www.json.org/json-en.html">JSON</a> (`j`); | ||
- The `<data>` **MUST** specify `application-id` equal to the Circulating Supply | ||
App ID. | ||
|
||
> **WARNING**: To preserve the existing ASA RBAC (e.g. Manager Address, Freeze Address, | ||
> etc.) it is necessary to **include all the existing role addresses** in the `AssetConfig`. | ||
> Not doing so would irreversibly disable the RBAC roles! | ||
|
||
### Example - JSON without version | ||
|
||
```text | ||
arc62:j{"application-id":123} | ||
``` | ||
|
||
## Reference Implementation | ||
|
||
> This section is non-normative. | ||
|
||
This section suggests a reference implementation of the Circulating Supply App. | ||
|
||
An Algorand-Python example is available [here](../assets/arc-0062). | ||
|
||
### Recommendations | ||
|
||
An ASA using the reference implementation **SHOULD NOT** assign the Reserve Address | ||
to the Circulating Supply App Account. | ||
|
||
A reference implementation **SHOULD** target a version of the AVM that supports | ||
foreign resources pooling (version 9 or greater). | ||
|
||
A reference implementation **SHOULD** use 3 external addresses, in addition to the | ||
Reserve Address, to define the not circulating supply. | ||
|
||
The **RECOMMENDED** labels for not-circulating balances are: `burned`, `locked` | ||
and `generic`. | ||
|
||
> To change the labels of not circulating addresses is sufficient to rename the | ||
> following constants just in `smart_contracts/circulating_supply/config.py`: | ||
> ```python | ||
> NOT_CIRCULATING_LABEL_1: Final[str] = "burned" | ||
> NOT_CIRCULATING_LABEL_2: Final[str] = "locked" | ||
> NOT_CIRCULATING_LABEL_3: Final[str] = "generic" | ||
> ``` | ||
|
||
### State Schema | ||
|
||
A reference implementation **SHOULD** allocate, at least, the following Global State | ||
variables: | ||
|
||
- `asset_id` as UInt64, initialized to `0` and set **only once** by the ASA Manager | ||
Address; | ||
- Not circulating address 1 (`burned`) as Bytes, initialized to the Global `Zero Address` | ||
and set by the ASA Manager Address; | ||
- Not circulating address 2 (`locked`) as Bytes, initialized to the Global `Zero Address` | ||
and set by the ASA Manager Address; | ||
- Not circulating address 3 (`generic`) as Bytes, initialized to the Global `Zero Address` | ||
and set by the ASA Manager Address. | ||
|
||
A reference implementation **SHOULD** enforce that, upon setting `burned`, `locked` | ||
and `generic` addresses, the latter already opted-in the `asset_id`. | ||
|
||
```json | ||
"state": { | ||
"global": { | ||
"num_byte_slices": 3, | ||
"num_uints": 1 | ||
}, | ||
"local": { | ||
"num_byte_slices": 0, | ||
"num_uints": 0 | ||
} | ||
}, | ||
"schema": { | ||
"global": { | ||
"declared": { | ||
"asset_id": { | ||
"type": "uint64", | ||
"key": "asset_id" | ||
}, | ||
"not_circulating_label_1": { | ||
"type": "bytes", | ||
"key": "burned" | ||
}, | ||
"not_circulating_label_2": { | ||
"type": "bytes", | ||
"key": "locked" | ||
}, | ||
"not_circulating_label_3": { | ||
"type": "bytes", | ||
"key": "generic" | ||
} | ||
}, | ||
"reserved": {} | ||
}, | ||
"local": { | ||
"declared": {}, | ||
"reserved": {} | ||
} | ||
}, | ||
``` | ||
|
||
### Circulating Supply Getter | ||
|
||
A reference implementation **SHOULD** enforce that the `asset_id` Global Variable | ||
is equal to the `asset_id` argument of the `arc62_get_circulating_supply` getter | ||
method. | ||
|
||
> Alternatively the reference implementation could ignore the `asset_id` argument | ||
> and use directly the `asset_id` Global Variable. | ||
|
||
A reference implementation **SHOULD** return the ASA _circulating supply_ as: | ||
|
||
```text | ||
circulating_supply = total - reserve_balance - burned_balance - locked_balance - generic_balance | ||
``` | ||
|
||
Where: | ||
|
||
- `total` is the total supply of the ASA (`asset_id`); | ||
- `reserve_balance` is the ASA balance hold by the Reserve Address or `0` if the | ||
address is set to the Global `ZeroAddress` or not opted-in `asset_id`; | ||
- `burned_balance` is the ASA balance hold by the Burned Address or `0` if the address | ||
is set to the Global `ZeroAddress` or is not opted-in `asset_id`; | ||
- `locked_balance` is the ASA balance hold by the Locked Address or `0` if the address | ||
is set to the Global `ZeroAddress` or not opted-in `asset_id`; | ||
- `generic_balance` is the ASA balance hold by a Generic Address or `0` if the address | ||
is set to the Global `ZeroAddress` or not opted-in `asset_id`. | ||
|
||
## Security Considerations | ||
|
||
Permissions over the Circulating Supply App setting and update **SHOULD** be granted | ||
to the ASA Manager Address. | ||
|
||
> The ASA trust-model (i.e. who sets the Reserve Address) is extended to the generalized | ||
> ASA circulating supply definition. | ||
|
||
## Copyright | ||
|
||
Copyright and related rights waived via <a href="https://creativecommons.org/publicdomain/zero/1.0/">CCO</a>. |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,52 @@ | ||
[algokit] | ||
min_version = "v2.0.0" | ||
|
||
[generate.smart-contract] | ||
description = "Generate a new smart contract for existing project" | ||
path = ".algokit/generators/create_contract" | ||
|
||
[generate.env-file] | ||
description = "Generate a new generic or Algorand network specific .env file" | ||
path = ".algokit/generators/create_env_file" | ||
|
||
[project] | ||
type = 'contract' | ||
name = 'asa-circulating-supply' | ||
artifacts = 'smart_contracts/artifacts' | ||
|
||
[project.deploy] | ||
command = "poetry run python -m smart_contracts deploy" | ||
environment_secrets = [ | ||
"DEPLOYER_MNEMONIC", | ||
"DISPENSER_MNEMONIC", | ||
] | ||
|
||
[project.deploy.localnet] | ||
environment_secrets = [] | ||
|
||
[project.run] | ||
# Commands intented for use locally and in CI | ||
build = { commands = [ | ||
'poetry run python -m smart_contracts build', | ||
], description = 'Build all smart contracts in the project' } | ||
test = { commands = [ | ||
'poetry run pytest', | ||
], description = 'Run smart contract tests' } | ||
audit = { commands = [ | ||
'poetry run pip-audit', | ||
], description = 'Audit with pip-audit' } | ||
lint = { commands = [ | ||
'poetry run black --check --diff .', | ||
'poetry run ruff check .', | ||
'poetry run mypy', | ||
], description = 'Perform linting' } | ||
audit-teal = { commands = [ | ||
# 🚨 IMPORTANT 🚨: For strict TEAL validation, remove --exclude statements. The default starter contract is not for production. Ensure thorough testing and adherence to best practices in smart contract development. This is not a replacement for a professional audit. | ||
'algokit task analyze smart_contracts/artifacts --recursive --force --exclude rekey-to --exclude is-updatable --exclude missing-fee-check --exclude is-deletable --exclude can-close-asset --exclude can-close-account --exclude unprotected-deletable --exclude unprotected-updatable', | ||
], description = 'Audit TEAL files' } | ||
|
||
# Commands intented for CI only, prefixed with `ci-` by convention | ||
ci-teal-diff = { commands = [ | ||
'git add -N ./smart_contracts/artifacts', | ||
'git diff --exit-code --minimal ./smart_contracts/artifacts', | ||
], description = 'Check TEAL files for differences' } |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
To enable the use case in which getter calls are made on-chain by another application (and not just simulated), would it be reasonable to include the following statement?
This also raises another question, which is: Should the ARC require that its implementations specify the minimum AVM version required to call the method?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this is an implementation detail. I would rather add a recommendation for the reference implementation to use an AVM version that supports resource pooling. The specification of the method interface does not really care about how resources are populated (e.g. they could be added to the AppCall directly or provided somehow within the same Group Transaction).
Also, maybe it is worth adding a non normative section in Usage to suggest that, if the method is supposed to be called as C2C call, then the implementation should provide details about foreign resources required to caller apps.