token.gno
6.71 Kb ยท 265 lines
1package grc20
2
3import (
4 "math/overflow"
5 "std"
6 "strconv"
7
8 "gno.land/p/demo/ufmt"
9)
10
11// NewToken creates a new Token.
12// It returns a pointer to the Token and a pointer to the Ledger.
13// Expected usage: Token, admin := NewToken("Dummy", "DUMMY", 4)
14func NewToken(name, symbol string, decimals uint) (*Token, *PrivateLedger) {
15 if name == "" {
16 panic("name should not be empty")
17 }
18 if symbol == "" {
19 panic("symbol should not be empty")
20 }
21 // XXX additional checks (length, characters, limits, etc)
22
23 ledger := &PrivateLedger{}
24 token := &Token{
25 name: name,
26 symbol: symbol,
27 decimals: decimals,
28 ledger: ledger,
29 }
30 ledger.token = token
31 return token, ledger
32}
33
34// GetName returns the name of the token.
35func (tok Token) GetName() string { return tok.name }
36
37// GetSymbol returns the symbol of the token.
38func (tok Token) GetSymbol() string { return tok.symbol }
39
40// GetDecimals returns the number of decimals used to get the token's precision.
41func (tok Token) GetDecimals() uint { return tok.decimals }
42
43// TotalSupply returns the total supply of the token.
44func (tok Token) TotalSupply() uint64 { return tok.ledger.totalSupply }
45
46// KnownAccounts returns the number of known accounts in the bank.
47func (tok Token) KnownAccounts() int { return tok.ledger.balances.Size() }
48
49// BalanceOf returns the balance of the specified address.
50func (tok Token) BalanceOf(address std.Address) uint64 {
51 return tok.ledger.balanceOf(address)
52}
53
54// Allowance returns the allowance of the specified owner and spender.
55func (tok Token) Allowance(owner, spender std.Address) uint64 {
56 return tok.ledger.allowance(owner, spender)
57}
58
59func (tok *Token) RenderHome() string {
60 str := ""
61 str += ufmt.Sprintf("# %s ($%s)\n\n", tok.name, tok.symbol)
62 str += ufmt.Sprintf("* **Decimals**: %d\n", tok.decimals)
63 str += ufmt.Sprintf("* **Total supply**: %d\n", tok.ledger.totalSupply)
64 str += ufmt.Sprintf("* **Known accounts**: %d\n", tok.KnownAccounts())
65 return str
66}
67
68// Getter returns a TokenGetter function that returns this token. This allows
69// storing indirect pointers to a token in a remote realm.
70func (tok *Token) Getter() TokenGetter {
71 return func() *Token {
72 return tok
73 }
74}
75
76// SpendAllowance decreases the allowance of the specified owner and spender.
77func (led *PrivateLedger) SpendAllowance(owner, spender std.Address, amount uint64) error {
78 if !owner.IsValid() {
79 return ErrInvalidAddress
80 }
81 if !spender.IsValid() {
82 return ErrInvalidAddress
83 }
84
85 currentAllowance := led.allowance(owner, spender)
86 if currentAllowance < amount {
87 return ErrInsufficientAllowance
88 }
89
90 key := allowanceKey(owner, spender)
91 newAllowance := currentAllowance - amount
92
93 if newAllowance == 0 {
94 led.allowances.Remove(key)
95 } else {
96 led.allowances.Set(key, newAllowance)
97 }
98
99 return nil
100}
101
102// Transfer transfers tokens from the specified from address to the specified to address.
103func (led *PrivateLedger) Transfer(from, to std.Address, amount uint64) error {
104 if !from.IsValid() {
105 return ErrInvalidAddress
106 }
107 if !to.IsValid() {
108 return ErrInvalidAddress
109 }
110 if from == to {
111 return ErrCannotTransferToSelf
112 }
113
114 var (
115 toBalance = led.balanceOf(to)
116 fromBalance = led.balanceOf(from)
117 )
118
119 if fromBalance < amount {
120 return ErrInsufficientBalance
121 }
122
123 var (
124 newToBalance = toBalance + amount
125 newFromBalance = fromBalance - amount
126 )
127
128 led.balances.Set(string(to), newToBalance)
129 led.balances.Set(string(from), newFromBalance)
130
131 std.Emit(
132 TransferEvent,
133 "from", from.String(),
134 "to", to.String(),
135 "value", strconv.Itoa(int(amount)),
136 )
137
138 return nil
139}
140
141// TransferFrom transfers tokens from the specified owner to the specified to address.
142// It first checks if the owner has sufficient balance and then decreases the allowance.
143func (led *PrivateLedger) TransferFrom(owner, spender, to std.Address, amount uint64) error {
144 if led.balanceOf(owner) < amount {
145 return ErrInsufficientBalance
146 }
147
148 // allowance must be sufficient
149 currentAllowance := led.allowance(owner, spender)
150 if currentAllowance < amount {
151 return ErrInsufficientAllowance
152 }
153
154 if err := led.Transfer(owner, to, amount); err != nil {
155 return err
156 }
157
158 // decrease the allowance only when transfer is successful
159 key := allowanceKey(owner, spender)
160 newAllowance := currentAllowance - amount
161
162 if newAllowance == 0 {
163 led.allowances.Remove(key)
164 } else {
165 led.allowances.Set(key, newAllowance)
166 }
167
168 return nil
169}
170
171// Approve sets the allowance of the specified owner and spender.
172func (led *PrivateLedger) Approve(owner, spender std.Address, amount uint64) error {
173 if !owner.IsValid() || !spender.IsValid() {
174 return ErrInvalidAddress
175 }
176
177 led.allowances.Set(allowanceKey(owner, spender), amount)
178
179 std.Emit(
180 ApprovalEvent,
181 "owner", string(owner),
182 "spender", string(spender),
183 "value", strconv.Itoa(int(amount)),
184 )
185
186 return nil
187}
188
189// Mint increases the total supply of the token and adds the specified amount to the specified address.
190func (led *PrivateLedger) Mint(address std.Address, amount uint64) error {
191 if !address.IsValid() {
192 return ErrInvalidAddress
193 }
194
195 // XXX: math/overflow is not supporting uint64.
196 // This checks prevents overflow but makes the totalSupply limited to a uint63.
197 sum, ok := overflow.Add64(int64(led.totalSupply), int64(amount))
198 if !ok {
199 return ErrOverflow
200 }
201
202 led.totalSupply = uint64(sum)
203 currentBalance := led.balanceOf(address)
204 newBalance := currentBalance + amount
205
206 led.balances.Set(string(address), newBalance)
207
208 std.Emit(
209 TransferEvent,
210 "from", "",
211 "to", string(address),
212 "value", strconv.Itoa(int(amount)),
213 )
214
215 return nil
216}
217
218// Burn decreases the total supply of the token and subtracts the specified amount from the specified address.
219func (led *PrivateLedger) Burn(address std.Address, amount uint64) error {
220 if !address.IsValid() {
221 return ErrInvalidAddress
222 }
223
224 currentBalance := led.balanceOf(address)
225 if currentBalance < amount {
226 return ErrInsufficientBalance
227 }
228
229 led.totalSupply -= amount
230 newBalance := currentBalance - amount
231
232 led.balances.Set(string(address), newBalance)
233
234 std.Emit(
235 TransferEvent,
236 "from", string(address),
237 "to", "",
238 "value", strconv.Itoa(int(amount)),
239 )
240
241 return nil
242}
243
244// balanceOf returns the balance of the specified address.
245func (led PrivateLedger) balanceOf(address std.Address) uint64 {
246 balance, found := led.balances.Get(address.String())
247 if !found {
248 return 0
249 }
250 return balance.(uint64)
251}
252
253// allowance returns the allowance of the specified owner and spender.
254func (led PrivateLedger) allowance(owner, spender std.Address) uint64 {
255 allowance, found := led.allowances.Get(allowanceKey(owner, spender))
256 if !found {
257 return 0
258 }
259 return allowance.(uint64)
260}
261
262// allowanceKey returns the key for the allowance of the specified owner and spender.
263func allowanceKey(owner, spender std.Address) string {
264 return owner.String() + ":" + spender.String()
265}