-
Notifications
You must be signed in to change notification settings - Fork 7
/
assets.ss
245 lines (218 loc) · 10.4 KB
/
assets.ss
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
;; Support for multiple classes of assets in a contract
;; TODO: add support on the client side in the same classes, too.
;; TODO: make that part of support of assets on multiple blockchains!
(export #t)
(import
:std/assert :std/format :std/iter
:std/misc/decimal :std/misc/hash :std/misc/list :std/misc/string
:std/parser/ll1
:std/srfi/1 :std/srfi/13
:std/sugar :std/text/char-set
:clan/base :clan/string
:clan/poo/object :clan/poo/brace
./assembly ./types ./ethereum ./abi ./evm-runtime ./network-config ./json-rpc ./erc20 ./simple-apps
./transaction ./tx-tracker)
;; TODO: rename asset to resource
;; for ERC721s, multiple resources in a resource-directory or resource-collection?
;; keys are uppercase symbols such as ETH, PET, CED, QASPET, RBTPET, etc.
(def asset-table (hash))
;; lookup-asset : Symbol -> AssetType
(def (lookup-asset s)
(hash-ref/default asset-table s
(lambda () (error 'lookup-asset s "not found in" (hash-keys asset-table)))))
;; register-asset! : AssetType -> Void
(def (register-asset! a) (hash-put! asset-table (.@ a .symbol) a))
;; Abstract interface for an asset type.
(define-type (Asset @ [Type.])
.element?: (lambda (v)
(and (object? v) (.has? v .symbol) (hash-key? asset-table (.@ v .symbol))))
.sexp<-: (lambda (a) `(lookup-asset ',(.@ a .symbol)))
.json<-: (lambda (a) (symbol->string (.@ a .symbol)))
.<-json: (lambda (j) (lookup-asset (string->symbol j)))
.string<-: (lambda (a) (symbol->string (.@ a .symbol)))
.<-string: (lambda (s) (lookup-asset (string->symbol s)))
.bytes<-: (lambda (a) (string->bytes (symbol->string (.@ a .symbol))))
.<-bytes: (lambda (b) (lookup-asset (string->symbol (bytes->string b))))
;; Implementations should additionally define:
;; Query the current balance of this asset for an address.
;;
;; .get-balance : @ <- .Address
;; (.transfer sender recipient amount) transfers 'amount' funds from 'sender' to
;; 'recipient'. Caller must be authorized to act on behalf of the sender.
;;
;; .transfer : <- .Address .Address @
;; (.commit-deposit! amount) generates EVM code to finalize/verify a deposit
;; of 'amount' into the consensus. This will be called once per asset type
;; at transaction commit.
;;
;; .commit-deposit! : (EVMThunk <-) <- (EVMThunk Amount <-)
;; (.commit-withdraw! recipient amount balance-var) is generates EVM code to
;; finalize/verify a withdrawal of 'amount' from the consensus. 'recipient' is the
;; participant making the withdrawal, and balance-var is the static variable holding
;; the balance for this (recipient, asset type) pair. called at transaction commit
;; once for each such pair.
;;
;; .commit-withdraw!: ;; (EVMThunk <-) <- (EVMThunk .Address <-) (EVMThunk @ <-) StaticVar
;; .commit-withdraw-all! is like .commit-withdraw!, but:
;;
;; - Instead of taking the participant as a (scheme) parameter, it is expected
;; to be at the top of the stack.
;; - It doesn't take an amount; instead, the entire balance is withdrawn.
;;
;; .commit-withdraw-all!: (EVMThunk <- .Address) <- StaticVar
;; (.approve-deposit! sender recipient amount) pre-approves a deposit into 'recipient'
;; from account 'sender', with the given amount, if necessary.
;;
;; .approve-deposit! : <- .Address .Address @
)
(define-type (TokenAmount @ [] .decimals .validate .symbol)
.denominator: (expt 10 .decimals)
;; NB: we use the US convention of currency symbol first, decimal amount second.
.string<-: (lambda (x) (format "~a ~a" .symbol (decimal->string (/ x .denominator))))
.<-string: (lambda (s)
(assert! (string-prefix? (format "~a " .symbol) s))
(.validate (*
.denominator
(string->decimal
s
sign-allowed?: #t
exponent-allowed: #t
;; TODO: should separators be taken from the user language environment?
group-separator: #\,
decimal-mark: #\.
start: (1+ (string-length (symbol->string .symbol))))))))
(define-type (Ether @ [TokenAmount UInt256] ;; or should it just be UInt96 ???
.length-in-bytes .length-in-bits)
.asset-code: 0
.network: 'eth
.name: "Ether"
.symbol: 'ETH
.decimals: 18
.Address: Address
.get-balance: ;; @ <- .Address
(lambda (address) (eth_getBalance address 'latest))
.transfer:
(lambda (sender recipient amount)
(post-transaction (transfer-tokens
from: sender
to: recipient
value: amount)))
;; NB: The above crucially depends on the end-of-transaction code including the below check,
;; that must be AND'ed with all other checks before [&require!]
.commit-deposit!: ;; (EVMThunk <-) <- (EVMThunk @ <-)
(lambda (amount)
(&begin amount CALLVALUE EQ &require!))
.commit-withdraw!: ;; (EVMThunk <-) <- (EVMThunk .Address <-) (EVMThunk @ <-) StaticVar
(lambda (recipient amount balance-var)
(&begin amount recipient DUP2 (&sub-var! balance-var) &send-ethers!)) ;; Transfer!
.commit-withdraw-all!:
(lambda (balance-var) ;; (EVMThunk <- .Address) <- StaticVar
(&begin (.@ balance-var get) SWAP1 &send-ethers! 0 (.@ balance-var set!)))
.approve-deposit!:
(lambda (sender recipient amount) (void)))
(register-asset! Ether)
(define-type (ERC20 @ [TokenAmount UInt256] ;; https://eips.ethereum.org/EIPS/eip-20
.contract-address ;; : Address
.name ;; : String ;; full name, e.g. "FooToken"
.symbol ;; : Symbol ;; symbol, typically a TLA, e.g. 'FOO
.decimals) ;; : Nat ;; number of decimals by which to divide the integer amount to get token amount
.asset-code: .contract-address
.Address: Address
.get-balance: ;; @ <- .Address
(lambda (address) (erc20-balance .contract-address address))
.transfer:
(lambda (sender recipient amount)
(erc20-transfer .contract-address sender recipient amount))
.commit-deposit!: ;; (EVMThunk <-) <- (EVMThunk @ <-)
(lambda (amount) ;; tmp@ is the constant offset to a 100-byte scratch buffer
(&begin
transferFrom-selector (&mstoreat/overwrite-after tmp100@ 4)
CALLER (&mstoreat (+ tmp100@ 4))
ADDRESS (&mstoreat (+ tmp100@ 36))
amount (&mstoreat (+ tmp100@ 68))
32 tmp100@ 100 DUP2 0 .contract-address GAS CALL
;; check that both the was successful and its boolean result true:
(&mloadat tmp100@) AND &require!))
.commit-withdraw!: ;; (EVMThunk <-) <- (EVMThunk .Address <-) (EVMThunk @ <-) StaticVar
(lambda (recipient amount balance-var)
(&begin
recipient
(&erc20-commit-withdraw
.contract-address
amount
balance-var)))
.commit-withdraw-all!:
(lambda (balance-var) ;; (EVMThunk <- .Address) <- StaticVar
(&erc20-commit-withdraw
.contract-address
(.@ balance-var get)
balance-var))
.approve-deposit!:
(lambda (sender recipient amount)
(erc20-approve .contract-address sender recipient amount)))
;; TODO: *if/when* we have a shared contract between multiple Glow interactions,
;; without a consensus on which ERC20 contracts can be trusted not to be exploit vectors,
;; then we need to add a flag to prevent re-entrancy (cost: ~5000 GAS) before we call out to
;; token contracts for withdrawals. Alternatively, if there's only one state variable at stake,
;; we can check that the state variable wasn't modified by a recursive call before we modify it,
;; (cost: ~700 gas? 2100?) which is slightly cheaper.
;; &erc20-commit-withdraw : (EVMThunk <- Address) <- Address (EVMThunk TokenAmount <-) StaticVar
;;
;; Sends funds for an erc20 token to a participant. Parameters:
;;
;; * contract-address is the address of the erc20 contract.
;; * amount pushes the amount to send
;; * balance-var is the balance variable to update.
;;
;; The resulting EVMThunk expects the participant address at the top of the stack.
(def (&erc20-commit-withdraw contract-address amount balance-var)
(&begin
transfer-selector (&mstoreat/overwrite-after tmp100@ 4)
;; recipient is on top of the stack already.
(&mstoreat (+ tmp100@ 4))
amount DUP1 (&sub-var! balance-var) (&mstoreat (+ tmp100@ 36))
32 tmp100@ 68 DUP2 0 contract-address GAS CALL
;; check that both the call was successful and that its boolean result was true:
(&mloadat tmp100@) AND &require!))
(def ll1-asset-amount
(ll1* cons (ll1-char+ char-ascii-alphabetic?) ;; asset name
;; TODO: programmable group-separator?
(ll1-begin ll1-skip-space* (cut ll1-decimal <> group-separator: #\,))))
(def (asset-amount<-string string trim-spaces?: (trim-spaces? #t))
(ll1/string ll1-asset-amount
(if trim-spaces? (string-trim-spaces string) string)
"asset-amount"))
(def (display-asset-amount asset-amount port)
(with ([asset . amount] asset-amount)
(display asset port) (write-char #\space port) (write-decimal amount port)))
(def (string<-asset-amount asset-amount)
(call-with-output-string (cut display-asset-amount asset-amount <>)))
(def (asset->network a)
(hash-ref ethereum-networks (symbol->string (.@ a .network))))
;; native-asset? : Bool <- Asset
;; Produces true if `a` is the native asset of its associated network
(def (native-asset? a)
(def network (asset->network a))
(def native-name (.@ network nativeCurrency symbol))
(equal? (.@ a .symbol) native-name))
;; lookup-native-asset : Asset <- EthereumConfig
;; Produces the native asset of the network from (ethereum-config)
(def (lookup-native-asset (ec (ethereum-config)))
(lookup-asset (.@ ec nativeCurrency symbol)))
;; find-network-assets : Network -> [Listof Asset]
(def (find-network-assets (network (ethereum-config)))
(for/collect ((p (hash->list/sort asset-table symbol<?))
when (equal? (asset->network (cdr p)) network))
(cdr p)))
;; Add Native Currencies of ethereum-networks to asset table.
(def (register-native-asset _ network)
(def nativeCurrency (.@ network nativeCurrency))
(hash-ensure-ref asset-table
(.@ nativeCurrency symbol)
(lambda ()
{(:: @ Ether)
.name: (.@ nativeCurrency name)
.symbol: (.@ nativeCurrency symbol)
.decimals: (.@ nativeCurrency decimals)
.network: (string->symbol (.@ network shortName))})))
(hash-for-each register-native-asset ethereum-networks)