The Convex Asset Model is a universal system for expressing and controlling digital assets in an on-chain environment. The key motivation for the Asset Model is to enable economic value transfer using digital assets: digital assets can be securely owned, traded and used as part of contractual agreements just like real world assets. Of course, the security of the digital assets is enforced by the security guarantees of the Convex network.
A key design goal is therefore that the API is universal and extensible, in the sense that it can be used to handle a diverse ecosystem of digital assets, including asset types that have not yet been invented. By way of example, it should be possible to use the same API to transfer a quantity of a fungible token:
(asset/transfer destination-address [currency.USD 10000] )
As it is to transfer a set of numbered NFTs:
(asset/transfer destination-address [asset.nfts #{101 102 105}] )
So... why is this important?
- It reduces the number of APIs that developers must learn: instead of having separate APIs for fungible token, NFTs, game items etc. we offer a single unified API.
- It makes it possible to write generic smart contracts that can work with any type of asset.
- It makes it possible to innovate with new types of asset without having to redesign user code
We therefore propose the following objectives:
- The Asset Model should be able to express all types of on-chain digital assets (fungible tokens, security tokens, stablecoins, NFTs, voting rights etc.)
- There should be a standard API for users that works with all types of assets in a generic way
- There should be a standard SPI that allows for flexibility in underlying asset implementations - in particular it should be possible to create new kinds of digital assets and new rules / behaviours without changing the user level API
The Asset Model should allow efficient and simple implementations to minimise transaction costs and memory usage. If digital assets are to be widely accepted as a part of economic value exchange, it is essential that they are efficient and offer low transaction costs compared to alternatives.
An asset is a logical entity that supports ownership and transfer according to the requirements in this CAD.
Examples:
- A non-fungible token (NFT)
- A fungible token representing shares in a company
- A fungible token representing units of derivative "put" contract
Asset logic MUST be implemented by an actor on Convex. This actor may be referred to as the "asset implementation" or "asset actor".
An asset MAY map one-to-one to an actor, however a single actor MAY implement multiple assets. This allowance is primarily for efficiency reasons: if many assets share the same on-chain logic, it makes sense for a single actor to implement them all rather than deploying new actors for each one.
The use of an actor to provide the asset implementation is important for two reasons:
- It allows for the development of new types of compatible assets: these simply need to provide a new implementation and they can be used according to the standard CAD19 asset model, often without needing to change existing code.
- Actors allow for trusted code execution and governance, providing assurance that digital assets will behave correctly and not present unacceptable security risks
An asset path is a descriptor that identifies an asset. Asset paths are important because they enable a stable reference to a specific asset under consideration.
An asset path MUST be either:
- The address of the actor that provides the asset implementation - e.g. -
#1234
is a valid asset path referring to the asset implemented by the actor at the address#1234
- A vector where the first element is the address of the actor, and the second element of the vector is a scoped value interpreted by that actor on an implementation defined basis. e.g.
[#2345 :foo]
. The asset actor will see the scoped value as*scope*
when called.
A Vector-based asset path SHOULD be used to allow a single actor to implement many different digital assets, e.g.
- Currencies might be designated by an asset path like
[#123456 :USD]
- Derivative contracts such as put options might have an asset path that includes the underlying asset, strike price and expiry time e.g.
[#98765 [[#12345 :USD] 12500 1741948885345]]
- Bets on a football match might specify the match date and selected winner
[#8978 ["2023-6-06" "Manchester United"]]
(note that a bet on a different outcome of the same match would be a different fungible asset since they are not mutually fungible)
All Assets MUST define a notion of "quantity" that can be used to represent an amount or subdivision of an asset.
Because we want to enable innovation in the types and representations of assets, we do not restrict the definition of quantity to a specific type (e.g. integer amounts) - instead we define the mathematical properties that quantities must obey.
Quantities for any asset MUST be a commutative monoid (in the mathematical sense). This requirement is necessary in order for addition, subtraction and comparison of asset quantities to behave in well-defined ways.
Assets MUST define an addition function enable quantities to be additively combined, i.e. given any two valid quantities of an asset it MUST be possible to use the addition function to compute the total quantity. This is equivalent to the addition function of the commutative monoid.
Assets MUST define a comparison function enabling quantities to compared, i.e. given any two quantities of an asset it MUST be possible to use the comparison function to determine if one quantity is a subset of the other. This is equivalent to the algebraic pre-ordering of the monoid.
Assets MUST define a subtraction function enabling quantities to be subtracted, i.e. given two quantities of an asset where the first is "larger" than the second (as defined by the comparison function), it MUST be possible to subtract the second value from the first and get a result that is also a valid quantity.
Assets MUST define a zero quantity that logically represents an empty holding of an asset. This zero value is the identity element of the commutative monoid. The zero quantity will usually be an "empty" value such as #{}
or 0
.
The zero quantity SHOULD be the default balance of all accounts. Logically, an account should have a zero holding of an asset until some quantity of the asset is otherwise obtained. An example of an exception to this might be an asset implementation that gives a free non-zero quantity of the asset to all accounts, though this is probably unwise given the obvious potential for abuse.
For any given Asset, it MUST be possible to identify a Holding of the Asset for a given Account, where the Holding is the Quantity of the Asset that the Account owns.
An asset implementation MAY permit quantities to be specified using non-canonical values, provided that it MUST behave as if the equivalent canonical quantity was provided. For example, a NFT actor could reasonably consider a non-set value such as 123
to refer to the singleton set #{123}
representing a single numbered NFT.
An asset implementation MAY attempt to produce reasonable results where quantity operations are passed arguments that are not normally valid, e.g. subtracting 3000
from 1000
would reasonably produce the result -2000
, even though this is not a valid balance. However users of assets SHOULD NOT rely on this behaviour (in this case, subtraction is invalid because the first argument is "less than" the amount subtracted).
When passed as an argument to an asset implementation, the value nil
MUST be treated as the zero quantity. This requirement ensures that a zero-equivalent value is known for all implementations, and can be used by generic code without having incurring the cost of explicitly querying the zero value.
- Fungible tokens typically use a quantity expressed as non-negative integers e.g.
0
,1000
,987654321
- Non-fungible tokens typically use a quantity expressed as sets of NFT IDs e.g.
#{101 1002 1003}
- An asset representing a voting right may use a boolean quantity
true
andfalse
Quantities of digital assets are owned by accounts in Convex.
Assets may be owned by either user accounts or actors. In the latter case, it should be expected that the actor implements code able to manage the assets that it owns.
Assets SHOULD be transferable, i.e. it should be possible for a quantity of an asset to be transferred from one owned to another.
It is often necessary for a smart contract to ensure that it receives another asset before it takes some action: for example a contract for sale would expect payment to be made before allowing the purchased assets to be released.
A typical process would be something like:
- Account
A
offers a quantity of assetF
to accountB
(whereB
is a smart contract) A
calls a callable smart contract function onB
to request a transactionB
checks preconditions for the transaction as necessaryB
accepts the quantity ofF
from the caller (A
) to pay for the transaction- Assuming all is successful,
B
completes the transaction B
returns to caller with transaction complete- Optional:
A
closes down the offer (only relevant if some non-accepted quantity remains)
Assets SHOULD implement a system of offers, whereby an owner may offer a quantity of an asset to another account, which can subsequently be accepted by that account.
Assets SHOULD implement a system of acceptance, whereby an account that has been offered a quantity of the asset may accept that quantity (or a partial quantity thereof).
Offers SHOULD remain open at the discretion of the offering account. However, closing of offers by a trusted 3rd party (e.g. to mitigate against security risks) MAY be acceptable in some cases.
The user API for the Asset Model is provided by the library convex.asset
. Examples below assume the user has imported an alias for convex.asset
library e.g.
(import convex.asset :as asset)
Users do not need to use the convex.asset
library to work with Convex digital assets - they are free to access the underlying actor functions directly. However the user API presents a convenient, well-tested interface that should be suitable for most purposes.
:::warning
convex.asset
is a library, which means that functions run in the context of the account that invokes them. Only do this with a trusted library, i.e. be sure you have imported the correct library. The address of the standard convex.asset
is #65
on Protonet.
:::
The balance
function gets the total quantity of an asset currently held by an account.
(asset/balance some-fungible-asset *address*)
=> 1500
;; Omitting the address argument is equivalent to using *address*
(asset/balance some-fungible-asset)
=> 1500
The returned value should always be a valid quantity for the specified asset. In particular, the owner should be able to transfer
this amount to send their entire holdings of the specified asset to another account.
If the address argument is omitted, the balance for the current account is queried (i.e. an implicit *address*
argument is used to specify the current account).
The transfer
function transfers a quantity of an asset to a recipient.
;; This transfers to account #13 1000 units of MY_TOKEN
(asset/transfer #13 [MY-TOKEN 1000])
As an alternative form, transfer
may be called with an asset path and quantity as separate arguments:
;; This transfers to account #13 1000 units of MY_TOKEN
(asset/transfer #13 MY-TOKEN 1000)
The receiving account MAY implement a ^:callable receive-asset
function, in which case instead of directly transferring the asset, the asset will be offered and receive-asset
will be called. Actors could, for example, use this so that they can automatically reject assets that they are not supposed to receive, and accept
assets that they can handle.
transfer
MUST fail if the caller does not have the quantity of assets specified. This SHOULD be a :FUNDS
error, since this is conventionally used to indicate the lack of a sufficient asset quantity.
Making an offer
makes a quantity of an asset available for a subsequent accept
by a nominated receiver.
(asset/offer receiver [fungible-token-address 1000])
:::warning If executed, an offer stays open until explicitly closed (by setting the quantity to zero). Users are advised to be cautious about making offers unless they expect the offer to be immediately accepted on terms they approve of (or rolled back in case of error) since the receiver may accept the offer at some arbitrary point in the future. :::
Asset implementations SHOULD reclaim memory by removing any offer records if the offer is set to the zero quantity. Failure to do so may result in wasted memory allowances.
The current offer of an asset from any sender to any receiver can be obtained with the get-offer
function:
(asset/get-offer fungible-token-address sender receiver)
Asset implementations MUST return the current offer, i.e. the quantity last specified via offer
from the sender to the receiver minus any quantities subsequently accepted with accept
.
Asset implementations MUST return the zero value if an offer does not exist.
An offer may be accepted by a receiver as follows:
(asset/accept sender [fungible-token-address 1000])
Often, the sender will be the *caller*
to an actor function which needs to take a quantity of an asset from the caller, e.g. as part of an atomic value exchange or as a payment of fees.
Asset acceptance MUST fail with a :FUNDS
error if the quantity could not be accepted because it exceeds the sender's current balance.
If accepted, the asset implementation MUST:
- Transfer the specified quantity of the asset from the sender to the receiver
- Subtract the accepted quantity from the offer (which may result in it becoming zero)
Tests whether a given address owns the specified quantity of an asset:
(asset/owns? #1456 fungible-token-address 1000)
;; alternative argument layout with an asset vector
(asset/owns? #1456 [fungible-token-address 1000])
The owns?
function MUST return true if and only if the balance of the owner is at least as large as the specified quantity, i.e. the owner would normally be able to transfer
this amount.
The total-supply
function obtains the current total supply of an asset, if available.
(asset/total-supply my-token)
=> 1000000000000000000
The result (if available) MUST equal the sum of all current balances of the asset.
The total-supply
function MAY return nil
if the asset does not support computation of the total supply.
The total supply of an asset MAY change during the lifetime of an asset, e.g. because new quantities are minted or burned.
Asset implementations SHOULD cache the total supply by keeping track of the total quantity of an asset created or destroyed, e.g. via mint and burn operations. This allows the total supply to be provided without adding up all current balances, which is likely to be unreasonably expensive for callers.
Assets are implemented by actors in the Convex asset model, and as such there are a number of issues that may arise if untrusted assets are used.
The general recommendation is that users SHOULD NOT interact with untrusted assets.
However, in some circumstances, it may be necessary to write code that may interact with untrusted assets. An example of this could be an "Auction House" actor that enables users to post lots containing other digital assets for auction. There is no way, in advance, for the auction house actor to know whether assets that may be sold in the future will be trusted or not. Hence the auction house must be written in a way that is robust to the inclusion of untrusted assets.
When an actor implementing an asset is interacted with, it may execute arbitrary code. Unless the actor is trusted, caution must be taken to mitigate these risks (as with a call to an untrusted actor).
While the Convex security model prevents the actor from directly taking actions on behalf of the caller (e.g. it cannot steal arbitrary assets) it may take other actions during the scope of the transaction. Developers interacting with untrusted assets should be particularly aware of:
- Re-entrancy attacks where a malicious asset calls back into the same smart contract
- Possibility that a malicious asset implementation may call other smart contracts, e.g. DEX exchange calls
If an asset implementation is untrusted, it is possible that the asset may not behave correctly according to the requirements in this CAD. Some examples:
- A transfer may appear to have succeeded, when it has in fact failed
- A malicious asset may lie about balances or the success or otherwise of transfers
- A 3rd party may have the ability to update holdings unilaterally, without the authorisation of owners.
- A previously well-behaved asset may be "upgraded" to become malicious
It is possible to transfer assets to an account that may be locked or otherwise inaccessible (e.g. a user account with a lost key pair). In such cases, the quantity of asset transferred may be irretrievably lost.
Some mitigations for this risk:
- Prefer solutions where the destination account explicitly calls
accept
to obtain the asset. By requiring the active participation of the receiver, this minimises the risk of the asset being locked in an inaccessible account. - Only implement
receive-asset
for an actor if the actor provides mechanisms for extracting assets at a later time. Otherwise,atransfer
to an actor risks losing access to the asset. - Perform off-chain validation that the destination user has access to their account (e.g. requiring a signature of a random number to prove possession of the appropriate key pair).
- Use appropriate governance mechanisms (e.g.
set-controller
) to enable account recovery as a last resort.
Asset implementations should ensure that they do not allow bugs resulting from numerical overflow or other issues relating to calculations of quantities. In general:
- Limits MUST be placed on total quantities available where overflow might otherwise be possible, e.g. a maximum supply for a fungible token. This limit should normally be enforced when minting / issuing new quantities.
- Functions involving quantities as arguments MUST check whether the quantity is valid, e.g. a non-negative integer less than or equal to the balance of the account requesting the transfer for a fungible token. Typically a check such as
(<= 0 amount balance)
is appropriate.
An asset implementation which is vulnerable to quantity overflow issues should be considered as broken (and therefore untrusted).
It is possible that some asset implementations (either by mistake or ill intention) are not compliant with the Convex Asset Model.
Users SHOULD NOT interact with non-compliant assets.
Some specific reasons for non-comliance are listed below:
Some assets may have balances / quantities that may vary independently of usage via the asset model. Some examples:
- An interest-bearing asset that pays out some form of periodic rewards that increase balance
- A utility token that has quantity deducted automatically through usage of services
Such assets are NOT COMPLIANT with the Convex Asset model and SHOULD NOT be implemented or used without extreme caution.
If applications nevertheless choose to make use of assets with volatile balances, applications SHOULD always check the balance at the start of any transaction rather than relying on any previously stored balances which may no longer be correct.
Such assets SHOULD NOT be used in general purpose smart contracts, since accounting for held balances may be invalidated.
If functionality is desired that requires balances to be varied, better approaches include:
- Requiring a user to explicitly claim interest or rewards
- Requiring balances to be deposited in a pre-payment account, so that usage quantities can be deducted and accounted for properly.
It is possible that multiple assets may refer to the same underlying quantity or resources. As such, code referencing multiple assets MUST NOT assume that operations on the assets are independent.
For example, transferring a quantity of an asset A
may affect the balance available of asset B
. This behaviour is NOT COMPLIANT for reasons similar to the volatile balances mentioned above
In general, it is safest to operate on each asset in turn - avoid interleaving actions API calls on multiple assets.