-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathdaoTokenDrop.py
327 lines (264 loc) · 12.8 KB
/
daoTokenDrop.py
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
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
import smartpy as sp
class DAOTokenDrop(sp.Contract):
"""This contract implements a DAO token drop distribution using a Merkle
tree.
The code is highly based on the Token Drop template by Anshu Jalan:
https://github.com/AnshuJalan/token-drop-template
The main modifications are:
- Possibility to update the Merkle tree.
- Introduction of a claim period.
- Introduction of a DAO treasury address that will receive the unclaimed
tokens after the claim period has passed.
- On-chain view to get an address claimed tokens.
"""
def __init__(self, administrator, metadata, token, treasury, merkle_root, expiration_date):
"""Initializes the contract.
"""
# Define the contract storage data types for clarity
self.init_type(sp.TRecord(
# The contract administrator
administrator=sp.TAddress,
# The contract metadata
metadata=sp.TBigMap(sp.TString, sp.TBytes),
# The DAO token address
token=sp.TAddress,
# The DAO treasury address
treasury=sp.TAddress,
# The Merkle tree root associated to the DAO distribution list
merkle_root=sp.TBytes,
# The claim period expiration date
expiration_date=sp.TTimestamp,
# The big map with the users that already claimed their tokens
claimed=sp.TBigMap(sp.TAddress, sp.TNat),
# The proposed new administrator address
proposed_administrator=sp.TOption(sp.TAddress)))
# Initialize the contract storage
self.init(
administrator=administrator,
metadata=metadata,
token=token,
treasury=treasury,
merkle_root=merkle_root,
expiration_date=expiration_date,
claimed=sp.big_map(),
proposed_administrator=sp.none)
# Fill the contract metadata
self.contract_metadata = {
"name": "Teia DAO token distribution contract",
"description": "Token distribution contract used for the Teia DAO",
"version": "1.0.0",
"authors": ["Teia Community <https://twitter.com/TeiaCommunity>"],
"homepage": "https://teia.art",
"source": {
"tools": ["SmartPy 0.16.0"],
"location": "https://github.com/teia-community/teia-smart-contracts/blob/main/python/contracts/daoTokenDrop.py"
},
"license": {
"name": "MIT",
"details": "The MIT License"
},
"interfaces": ["TZIP-016"],
"errors": [ {"error": {"string": "DROP_NOT_ADMIN"},
"expansion": {"string": "The account that executed the entry point is not the contract administrator"},
"languages": ["en"]},
{"error": {"string": "DROP_NO_NEW_ADMIN"},
"expansion": {"string": "The new administrator has not been proposed"},
"languages": ["en"]},
{"error": {"string": "DROP_NOT_PROPOSED_ADMIN"},
"expansion": {"string": "The operation can only be executed by the proposed administrator"},
"languages": ["en"]},
{"error": {"string": "DROP_TEZ_TRANSFER"},
"expansion": {"string": "The operation does not accept tez transfers"},
"languages": ["en"]},
{"error": {"string": "DROP_INVALID_MERKLE_PROOF"},
"expansion": {"string": "The provided Merkle proof is not valid"},
"languages": ["en"]},
{"error": {"string": "DROP_INVALID_LEAF"},
"expansion": {"string": "The provided leaf is not valid"},
"languages": ["en"]},
{"error": {"string": "DROP_SENDER_NOT_LEAF"},
"expansion": {"string": "The wallet that executed the operation is not the one in the leaf"},
"languages": ["en"]},
{"error": {"string": "DROP_ALL_TOKENS_CLAIMED"},
"expansion": {"string": "The wallet that executed the operation has already claimed all their tokens"},
"languages": ["en"]},
{"error": {"string": "DROP_CLAIM_EXPIRED"},
"expansion": {"string": "The token claim period has expired"},
"languages": ["en"]},
{"error": {"string": "DROP_CLAIM_NOT_EXPIRED"},
"expansion": {"string": "The token claim period has not expired"},
"languages": ["en"]}]}
self.init_metadata("contract_metadata", self.contract_metadata)
def check_is_administrator(self):
"""Checks that the address that called the entry point is the contract
administrator.
"""
sp.verify(sp.sender == self.data.administrator,
message="DROP_NOT_ADMIN")
def verify_proof(self, proof, leaf):
"""Computes the Merkle tree root from the provided proof and leaf and
checks that it coincides with the stored merkle root.
"""
# Loop over the proof elements and calculate the combined hash
combined_hash = sp.local("combined_hash", sp.sha256(leaf))
with sp.for_("proof_element", proof) as proof_element:
with sp.if_(combined_hash.value < proof_element):
combined_hash.value = sp.sha256(
combined_hash.value + proof_element)
with sp.else_():
combined_hash.value = sp.sha256(
proof_element + combined_hash.value)
# Check that the combined hash coincides with the stored Merkle root
sp.verify(combined_hash.value == self.data.merkle_root,
message="DROP_INVALID_MERKLE_PROOF")
def dao_transfer(self, address, amount):
"""Transfers some DAO tokens from the contract to an address.
"""
# Get a handle to the DAO token transfer entry point
token_transfer_handle = sp.contract(
t=sp.TList(sp.TRecord(
from_=sp.TAddress,
txs=sp.TList(sp.TRecord(
to_=sp.TAddress,
token_id=sp.TNat,
amount=sp.TNat).layout(
("to_", ("token_id", "amount"))))).layout(
("from_", "txs"))),
address=self.data.token,
entry_point="transfer").open_some()
# Execute the transfer
sp.transfer(
arg=sp.list([sp.record(
from_=sp.self_address,
txs=sp.list([sp.record(
to_=address,
token_id=sp.nat(0),
amount=amount)]))]),
amount=sp.mutez(0),
destination=token_transfer_handle)
@sp.entry_point
def claim(self, params):
"""Claims some DAO tokens.
"""
# Define the input parameter data type
sp.set_type(params, sp.TRecord(
proof=sp.TList(sp.TBytes),
leaf=sp.TBytes).layout(("proof", "leaf")))
# Check that the sender didn't transfer any tez
sp.verify(sp.amount == sp.tez(0), message="DROP_TEZ_TRANSFER")
# Check that the claim period didn't expire
sp.verify(sp.now < self.data.expiration_date,
message="DROP_CLAIM_EXPIRED")
# Unpack the leaf data
leaf_data_type = sp.TRecord(
address=sp.TAddress,
value=sp.TNat).layout(("address", "value"))
leaf_data = sp.compute(sp.unpack(
params.leaf, leaf_data_type).open_some("DROP_INVALID_LEAF"))
# Check that the sender coincides with the leaf address
sp.verify(sp.sender == leaf_data.address,
message="DROP_SENDER_NOT_LEAF")
# Check that the sender didn't claim all the tokens
unclaimed_tokens = sp.compute(sp.as_nat(
leaf_data.value - self.data.claimed.get(sp.sender, 0)))
sp.verify(unclaimed_tokens > 0, message="DROP_ALL_TOKENS_CLAIMED")
# Check that the provided proof is correct
self.verify_proof(params.proof, params.leaf)
# Transfer the unclaimed DAO token editions to the sender
self.dao_transfer(sp.sender, unclaimed_tokens)
# Update the claimed big map
self.data.claimed[sp.sender] = leaf_data.value
@sp.entry_point
def transfer_to_treasury(self, amount):
"""Transfers some DAO tokens to the DAO treasury.
This entrypoint can be executed by anyone only after the claim period
has expired.
"""
# Define the input parameter data type
sp.set_type(amount, sp.TNat)
# Check that the sender didn't transfer any tez
sp.verify(sp.amount == sp.tez(0), message="DROP_TEZ_TRANSFER")
# Check that the claim period has expired
sp.verify(sp.now > self.data.expiration_date,
message="DROP_CLAIM_NOT_EXPIRED")
# Transfer the DAO tokens to the treasury
self.dao_transfer(self.data.treasury, amount)
@sp.entry_point
def update_token(self, new_token):
"""Updates the DAO token address.
"""
# Define the input parameter data type
sp.set_type(new_token, sp.TAddress)
# Check that the administrator executed the entry point
self.check_is_administrator()
# Update the DAO token address
self.data.token = new_token
@sp.entry_point
def update_treasury(self, new_treasury):
"""Updates the DAO treasury address that will receive the unclaimed
tokens after the claim period.
"""
# Define the input parameter data type
sp.set_type(new_treasury, sp.TAddress)
# Check that the administrator executed the entry point
self.check_is_administrator()
# Update the DAO treasury address
self.data.treasury = new_treasury
@sp.entry_point
def update_merkle_root(self, new_merkle_root):
"""Updates the Merkle tree root to reflect an updated DAO token
distribution.
"""
# Define the input parameter data type
sp.set_type(new_merkle_root, sp.TBytes)
# Check that the administrator executed the entry point
self.check_is_administrator()
# Update the Merkle tree root
self.data.merkle_root = new_merkle_root
@sp.entry_point
def update_expiration_date(self, new_expiration_date):
"""Updates the claim expiration date.
"""
# Define the input parameter data type
sp.set_type(new_expiration_date, sp.TTimestamp)
# Check that the administrator executed the entry point
self.check_is_administrator()
# Update the claim expiration date
self.data.expiration_date = new_expiration_date
@sp.entry_point
def transfer_administrator(self, proposed_administrator):
"""Proposes to transfer the contract administrator to another address.
"""
# Define the input parameter data type
sp.set_type(proposed_administrator, sp.TAddress)
# Check that the administrator executed the entry point
self.check_is_administrator()
# Set the new proposed administrator address
self.data.proposed_administrator = sp.some(proposed_administrator)
@sp.entry_point
def accept_administrator(self):
"""The proposed administrator accepts the contract administrator
responsibilities.
"""
# Check that the proposed administrator executed the entry point
sp.verify(sp.sender == self.data.proposed_administrator.open_some(
"DROP_NO_NEW_ADMIN"), message="DROP_NOT_PROPOSED_ADMIN")
# Set the new administrator address
self.data.administrator = sp.sender
# Reset the proposed administrator value
self.data.proposed_administrator = sp.none
@sp.onchain_view(pure=True)
def claimed_tokens(self, address):
"""Returns the number of tokens claimed by the address.
"""
# Define the input parameter data type
sp.set_type(address, sp.TAddress)
# Return the number of claimed tokens
sp.result(self.data.claimed.get(address, 0))
sp.add_compilation_target("daoTokenDrop", DAOTokenDrop(
administrator=sp.address("tz1RssrimWo3B8TpCajiNjqBD3MfhUwEgxod"),
metadata=sp.utils.metadata_of_url("ipfs://QmVgckZRYgAMw6RqEHv4hZvJ1kqVYJ2qxRZQj6T6uFVTub"),
token=sp.address("KT1RJ6PbjHpwc3M5rw5s2Nbmefwbuwbdxton"),
treasury=sp.address("KT1J9FYz29RBQi1oGLw8uXyACrzXzV1dHuvb"),
merkle_root=sp.bytes("0x09b3bfe2615514519988b05eb69b9cb69383ed9e060bec3693a3fc8801dbf2b3"),
expiration_date=sp.timestamp_from_utc(2023, 11, 20, 23, 59, 59)))