NEP | Title | Author | Status | DiscussionsTo | Type | Category | Created | SupersededBy |
---|---|---|---|---|---|---|---|---|
21 |
Fungible Token Standard |
Evgeny Kuzyakov <[email protected]> |
Final |
Standards Track |
Contract |
29-Oct-2019 |
141 |
A standard interface for fungible tokens allowing for ownership, escrow and transfer, specifically targeting third-party marketplace integration.
NEAR Protocol uses an asynchronous sharded Runtime. This means the following:
Storage for different contracts and accounts can be located on the different shards. Two contracts can be executed at the same time in different shards. While this increases the transaction throughput linearly with the number of shards, it also creates some challenges for cross-contract development. For example, if one contract wants to query some information from the state of another contract (e.g. current balance), by the time the first contract receive the balance the real balance can change. It means in the async system, a contract can't rely on the state of other contract and assume it's not going to change.
Instead the contract can rely on temporary partial lock of the state with a callback to act or unlock, but it requires careful engineering to avoid dead locks.
In this standard we're trying to avoid enforcing locks, since most actions can still be completed without locks by transferring ownership to an escrow account.
Prior art:
ERC-20 standard NEP#4 NEAR NFT standard: #4 For latest lock proposals see Safes (#26)
We should be able to do the following:
- Initialize contract once. The given total supply will be owned by the given account ID.
- Get the total supply.
- Transfer tokens to a new user.
- Set a given allowance for an escrow account ID.
- Escrow will be able to transfer up this allowance from your account.
- Get current balance for a given account ID.
- Transfer tokens from one user to another.
- Get the current allowance for an escrow account on behalf of the balance owner. This should only be used in the UI, since a contract shouldn't rely on this temporary information.
There are a few concepts in the scenarios above:
- Total supply. It's the total number of tokens in circulation.
- Balance owner. An account ID that owns some amount of tokens.
- Balance. Some amount of tokens.
- Transfer. Action that moves some amount from one account to another account.
- Escrow. A different account from the balance owner who has permission to use some amount of tokens.
- Allowance. The amount of tokens an escrow account can use on behalf of the account owner.
Note, that the precision is not part of the default standard, since it's not required to perform actions. The minimum value is always 1 token.
Alice wants to send 5 wBTC tokens to Bob. Assumptions:
- The wBTC token contract is
wbtc
. - Alice's account is
alice
. - Bob's account is
bob
. - The precision on wBTC contract is
10^8
. - The 5 tokens is
5 * 10^8
or as a number is500000000
.
Alice needs to issue one transaction to wBTC contract to transfer 5 tokens (multiplied by precision) to Bob.
alice
callswbtc::transfer({"new_owner_id": "bob", "amount": "500000000"})
.
Alice wants to deposit 1000 DAI tokens to a compound interest contract to earn extra tokens. Assumptions:
- The DAI token contract is
dai
. - Alice's account is
alice
. - The compound interest contract is
compound
. - The precision on DAI contract is
10^18
. - The 1000 tokens is
1000 * 10^18
or as a number is1000000000000000000000
. - The compound contract can work with multiple token types.
Alice needs to issue 2 transactions. The first one to dai
to set an allowance for compound
to be able to withdraw tokens from alice
.
The second transaction is to the compound
to start the deposit process. Compound will check that the DAI tokens are supported and will try to withdraw the desired amount of DAI from alice
.
- If transfer succeeded,
compound
can increase local ownership foralice
to 1000 DAI - If transfer fails,
compound
doesn't need to do anything in current example, but maybe can notifyalice
of unsuccessful transfer.
alice
callsdai::set_allowance({"escrow_account_id": "compound", "allowance": "1000000000000000000000"})
.alice
callscompound::deposit({"token_contract": "dai", "amount": "1000000000000000000000"})
. During thedeposit
call,compound
does the following:- makes async call
dai::transfer_from({"owner_id": "alice", "new_owner_id": "compound", "amount": "1000000000000000000000"})
. - attaches a callback
compound::on_transfer({"owner_id": "alice", "token_contract": "dai", "amount": "1000000000000000000000"})
.
- makes async call
Charlie wants to exchange his wLTC to wBTC on decentralized exchange contract. Alex wants to buy wLTC and has 80 wBTC. Assumptions
- The wLTC token contract is
wltc
. - The wBTC token contract is
wbtc
. - The DEX contract is
dex
. - Charlie's account is
charlie
. - Alex's account is
alex
. - The precision on both tokens contract is
10^8
. - The amount of 9001 wLTC tokens is Alex wants is
9001 * 10^8
or as a number is900100000000
. - The 80 wBTC tokens is
80 * 10^8
or as a number is8000000000
. - Charlie has 1000000 wLTC tokens which is
1000000 * 10^8
or as a number is100000000000000
- Dex contract already has an open order to sell 80 wBTC tokens by
alex
towards 9001 wLTC. - Without Safes implementation, DEX has to act as an escrow and hold funds of both users before it can do an exchange.
Let's first setup open order by Alex on DEX. It's similar to Token deposit to a contract
example above.
- Alex sets an allowance on wBTC to DEX
- Alex calls deposit on Dex for wBTC.
- Alex calls DEX to make an new sell order.
Then Charlie comes and decides to fulfill the order by selling his wLTC to Alex on DEX. Charlie calls the DEX
- Charlie sets the allowance on wLTC to DEX
- Alex calls deposit on Dex for wLTC.
- Then calls DEX to take the order from Alex.
When called, DEX makes 2 async transfers calls to exchange corresponding tokens.
- DEX calls wLTC to transfer tokens DEX to Alex.
- DEX calls wBTC to transfer tokens DEX to Charlie.
alex
callswbtc::set_allowance({"escrow_account_id": "dex", "allowance": "8000000000"})
.alex
callsdex::deposit({"token": "wbtc", "amount": "8000000000"})
.dex
callswbtc::transfer_from({"owner_id": "alex", "new_owner_id": "dex", "amount": "8000000000"})
alex
callsdex::trade({"have": "wbtc", "have_amount": "8000000000", "want": "wltc", "want_amount": "900100000000"})
.charlie
callswltc::set_allowance({"escrow_account_id": "dex", "allowance": "100000000000000"})
.charlie
callsdex::deposit({"token": "wltc", "amount": "100000000000000"})
.dex
callswltc::transfer_from({"owner_id": "charlie", "new_owner_id": "dex", "amount": "100000000000000"})
charlie
callsdex::trade({"have": "wltc", "have_amount": "900100000000", "want": "wbtc", "want_amount": "8000000000"})
.dex
callswbtc::transfer({"new_owner_id": "charlie", "amount": "8000000000"})
dex
callswltc::transfer({"new_owner_id": "alex", "amount": "900100000000"})
The full implementation in Rust can be found there: https://github.com/nearprotocol/near-sdk-rs/blob/master/examples/fungible-token/src/lib.rs
NOTES:
- All amounts, balances and allowance are limited by U128 (max value
2**128 - 1
). - Token standard uses JSON for serialization of arguments and results.
- Amounts in arguments and results have are serialized as Base-10 strings, e.g.
"100"
. This is done to avoid JSON limitation of max integer value of2**53
.
Interface:
/******************/
/* CHANGE METHODS */
/******************/
/// Sets the `allowance` for `escrow_account_id` on the account of the caller of this contract
/// (`predecessor_id`) who is the balance owner.
pub fn set_allowance(&mut self, escrow_account_id: AccountId, allowance: U128);
/// Transfers the `amount` of tokens from `owner_id` to the `new_owner_id`.
/// Requirements:
/// * `amount` should be a positive integer.
/// * `owner_id` should have balance on the account greater or equal than the transfer `amount`.
/// * If this function is called by an escrow account (`owner_id != predecessor_account_id`),
/// then the allowance of the caller of the function (`predecessor_account_id`) on
/// the account of `owner_id` should be greater or equal than the transfer `amount`.
pub fn transfer_from(&mut self, owner_id: AccountId, new_owner_id: AccountId, amount: U128);
/// Transfer `amount` of tokens from the caller of the contract (`predecessor_id`) to
/// `new_owner_id`.
/// Act the same was as `transfer_from` with `owner_id` equal to the caller of the contract
/// (`predecessor_id`).
pub fn transfer(&mut self, new_owner_id: AccountId, amount: U128);
/****************/
/* VIEW METHODS */
/****************/
/// Returns total supply of tokens.
pub fn get_total_supply(&self) -> U128;
/// Returns balance of the `owner_id` account.
pub fn get_balance(&self, owner_id: AccountId) -> U128;
/// Returns current allowance of `escrow_account_id` for the account of `owner_id`.
///
/// NOTE: Other contracts should not rely on this information, because by the moment a contract
/// receives this information, the allowance may already be changed by the owner.
/// So this method should only be used on the front-end to see the current allowance.
pub fn get_allowance(&self, owner_id: AccountId, escrow_account_id: AccountId) -> U128;
- Current interface doesn't have minting, precision (decimals), naming. But it should be done as extensions, e.g. a Precision extension.
- It's not possible to exchange tokens without transferring them to escrow first.
- It's not possible to transfer tokens to a contract with a single transaction without setting the allowance first.
It should be possible if we introduce
transfer_with
function that transfers tokens and calls escrow contract. It needs to handle result of the execution and contracts have to be aware of this API.
- Support for multiple token types
- Minting and burning
- Precision, naming and short token name.
Copyright and related rights waived via CC0.