-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathETHCrowdfundBase.sol
399 lines (354 loc) · 16.7 KB
/
ETHCrowdfundBase.sol
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
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
// SPDX-License-Identifier: GPL-3.0
pragma solidity 0.8.20;
import "../utils/LibAddress.sol";
import "../utils/LibSafeCast.sol";
import "../party/Party.sol";
import "../gatekeepers/IGateKeeper.sol";
contract ETHCrowdfundBase is Implementation {
using LibRawResult for bytes;
using LibSafeCast for uint256;
using LibAddress for address payable;
enum CrowdfundLifecycle {
// In practice, this state is never used. If the crowdfund is ever in
// this stage, something is wrong (e.g. crowdfund was never initialized).
Invalid,
// Ready to accept contributions to reach contribution targets
// until a deadline or the minimum contribution target is reached and
// host finalizes.
Active,
// Expired and the minimum contribution target was not reached.
Lost,
// The crowdfund has expired and reached the minimum contribution
// target. It is now ready to finalize.
Won,
// A won crowdfund has been finalized, with funds transferred to the
// party and voting power successfully updated.
Finalized
}
// Options to be passed into `initialize()` when the crowdfund is created.
struct ETHCrowdfundOptions {
Party party;
address payable initialContributor;
address initialDelegate;
uint96 minContribution;
uint96 maxContribution;
bool disableContributingForExistingCard;
uint96 minTotalContributions;
uint96 maxTotalContributions;
uint16 exchangeRateBps;
uint16 fundingSplitBps;
address payable fundingSplitRecipient;
uint40 duration;
IGateKeeper gateKeeper;
bytes12 gateKeeperId;
}
error WrongLifecycleError(CrowdfundLifecycle lc);
error NotAllowedByGateKeeperError(
address contributor,
IGateKeeper gateKeeper,
bytes12 gateKeeperId,
bytes gateData
);
error OnlyPartyHostError();
error OnlyPartyDaoError(address notDao);
error OnlyPartyDaoOrHostError(address notDao);
error NotOwnerError(uint256 tokenId);
error OnlyWhenEmergencyActionsAllowedError();
error InvalidDelegateError();
error NotEnoughContributionsError(uint96 totalContribution, uint96 minTotalContributions);
error MinGreaterThanMaxError(uint96 min, uint96 max);
error MaxTotalContributionsCannotBeZeroError(uint96 maxTotalContributions);
error BelowMinimumContributionsError(uint96 contributions, uint96 minContributions);
error AboveMaximumContributionsError(uint96 contributions, uint96 maxContributions);
error InvalidExchangeRateError(uint16 exchangeRateBps);
error ContributingForExistingCardDisabledError();
error ZeroVotingPowerError();
error FundingSplitAlreadyPaidError();
error FundingSplitNotConfiguredError();
error InvalidMessageValue();
event Contributed(
address indexed sender,
address indexed contributor,
uint256 amount,
address delegate
);
event Finalized();
event FundingSplitSent(address indexed fundingSplitRecipient, uint256 amount);
event EmergencyExecuteDisabled();
event EmergencyExecute(address target, bytes data, uint256 amountEth);
// The `Globals` contract storing global configuration values. This contract
// is immutable and it’s address will never change.
IGlobals private immutable _GLOBALS;
/// @notice The address of the `Party` contract instance associated
/// with the crowdfund.
Party public party;
/// @notice The minimum amount of ETH that a contributor can send to
/// participate in the crowdfund.
uint96 public minContribution;
/// @notice The maximum amount of ETH that a contributor can send to
/// participate in the crowdfund per address.
uint96 public maxContribution;
/// @notice A boolean flag that determines whether contributors are allowed
/// to increase the voting power of their existing party cards.
bool public disableContributingForExistingCard;
/// @notice Whether the funding split has been claimed by the funding split
/// recipient.
bool public fundingSplitPaid;
/// @notice Whether the DAO has emergency powers for this crowdfund.
bool public emergencyExecuteDisabled;
/// @notice The minimum amount of total ETH contributions required for the
/// crowdfund to be considered successful.
uint96 public minTotalContributions;
/// @notice The maximum amount of total ETH contributions allowed for the
/// crowdfund.
uint96 public maxTotalContributions;
/// @notice The total amount of ETH contributed to the crowdfund so far.
uint96 public totalContributions;
/// @notice The timestamp at which the crowdfund will end or ended. If 0, the
/// crowdfund has finalized.
uint40 public expiry;
/// @notice The exchange rate to use for converting ETH contributions to
/// voting power in basis points (e.g. 10000 = 1:1).
uint16 public exchangeRateBps;
/// @notice The portion of contributions to send to the funding recipient in
/// basis points (e.g. 100 = 1%).
uint16 public fundingSplitBps;
/// @notice The address to which a portion of the contributions is sent to.
address payable public fundingSplitRecipient;
/// @notice The gatekeeper contract used to restrict who can contribute to the party.
IGateKeeper public gateKeeper;
/// @notice The ID of the gatekeeper to use for restricting contributions to the party.
bytes12 public gateKeeperId;
/// @notice The address a contributor is delegating their voting power to.
mapping(address => address) public delegationsByContributor;
// Set the `Globals` contract.
constructor(IGlobals globals) {
_GLOBALS = globals;
}
// Initialize storage for proxy contracts, credit initial contribution (if
// any), and setup gatekeeper.
function _initialize(ETHCrowdfundOptions memory opts) internal {
// Set the minimum and maximum contribution amounts.
if (opts.minContribution > opts.maxContribution) {
revert MinGreaterThanMaxError(opts.minContribution, opts.maxContribution);
}
minContribution = opts.minContribution;
maxContribution = opts.maxContribution;
// Set the min total contributions.
if (opts.minTotalContributions > opts.maxTotalContributions) {
revert MinGreaterThanMaxError(opts.minTotalContributions, opts.maxTotalContributions);
}
minTotalContributions = opts.minTotalContributions;
// Set the max total contributions.
if (opts.maxTotalContributions == 0) {
// Prevent this because when `maxTotalContributions` is 0 the
// crowdfund is invalid in `getCrowdfundLifecycle()` meaning it has
// never been initialized.
revert MaxTotalContributionsCannotBeZeroError(opts.maxTotalContributions);
}
maxTotalContributions = opts.maxTotalContributions;
// Set the party crowdfund is for.
party = opts.party;
// Set the crowdfund start and end timestamps.
expiry = uint40(block.timestamp + opts.duration);
// Set the exchange rate.
if (opts.exchangeRateBps == 0) revert InvalidExchangeRateError(opts.exchangeRateBps);
exchangeRateBps = opts.exchangeRateBps;
// Set the funding split and its recipient.
fundingSplitBps = opts.fundingSplitBps;
fundingSplitRecipient = opts.fundingSplitRecipient;
// Set whether to disable contributing for existing card.
disableContributingForExistingCard = opts.disableContributingForExistingCard;
}
/// @notice Get the current lifecycle of the crowdfund.
function getCrowdfundLifecycle() public view returns (CrowdfundLifecycle lifecycle) {
if (maxTotalContributions == 0) {
return CrowdfundLifecycle.Invalid;
}
uint256 expiry_ = expiry;
if (expiry_ == 0) {
return CrowdfundLifecycle.Finalized;
}
if (block.timestamp >= expiry_) {
if (totalContributions >= minTotalContributions) {
return CrowdfundLifecycle.Won;
} else {
return CrowdfundLifecycle.Lost;
}
}
return CrowdfundLifecycle.Active;
}
function _processContribution(
address payable contributor,
address delegate,
uint96 amount
) internal returns (uint96 votingPower) {
address oldDelegate = delegationsByContributor[contributor];
if (msg.sender == contributor || oldDelegate == address(0)) {
// Update delegate.
delegationsByContributor[contributor] = delegate;
} else {
// Prevent changing another's delegate if already delegated.
delegate = oldDelegate;
}
emit Contributed(msg.sender, contributor, amount, delegate);
// OK to contribute with zero just to update delegate.
if (amount == 0) return 0;
// Only allow contributions while the crowdfund is active.
CrowdfundLifecycle lc = getCrowdfundLifecycle();
if (lc != CrowdfundLifecycle.Active) {
revert WrongLifecycleError(lc);
}
// Check that the contribution amount is at or below the maximum.
uint96 maxContribution_ = maxContribution;
if (amount > maxContribution_) {
revert AboveMaximumContributionsError(amount, maxContribution_);
}
uint96 newTotalContributions = totalContributions + amount;
uint96 maxTotalContributions_ = maxTotalContributions;
if (newTotalContributions >= maxTotalContributions_) {
totalContributions = maxTotalContributions_;
// Finalize the crowdfund.
// This occurs before refunding excess contribution to act as a
// reentrancy guard.
_finalize(maxTotalContributions_);
// Refund excess contribution.
uint96 refundAmount = newTotalContributions - maxTotalContributions;
if (refundAmount > 0) {
amount -= refundAmount;
payable(msg.sender).transferEth(refundAmount);
}
} else {
totalContributions = newTotalContributions;
}
// Check that the contribution amount is at or above the minimum. This
// is done after `amount` is potentially reduced if refunding excess
// contribution. There is a case where this prevents a crowdfunds from
// reaching `maxTotalContributions` if the `minContribution` is greater
// than the difference between `maxTotalContributions` and the current
// `totalContributions`. In this scenario users will have to wait until
// the crowdfund expires or a host finalizes after
// `minTotalContribution` has been reached by calling `finalize()`.
uint96 minContribution_ = minContribution;
if (amount < minContribution_) {
revert BelowMinimumContributionsError(amount, minContribution_);
}
// Subtract fee from contribution amount if applicable.
address payable fundingSplitRecipient_ = fundingSplitRecipient;
uint16 fundingSplitBps_ = fundingSplitBps;
if (fundingSplitRecipient_ != address(0) && fundingSplitBps_ > 0) {
// Removes funding split from contribution amount in a way that
// avoids rounding errors for very small contributions <1e4 wei.
amount = (amount * (1e4 - fundingSplitBps_)) / 1e4;
}
// Calculate voting power.
votingPower = (amount * exchangeRateBps) / 1e4;
if (votingPower == 0) revert ZeroVotingPowerError();
}
/// @notice Calculate the contribution amount from the given voting power.
/// @param votingPower The voting power to convert to a contribution amount.
/// @return amount The contribution amount.
function convertVotingPowerToContribution(
uint96 votingPower
) public view returns (uint96 amount) {
amount = (votingPower * 1e4) / exchangeRateBps;
// Add back funding split to contribution amount if applicable.
address payable fundingSplitRecipient_ = fundingSplitRecipient;
uint16 fundingSplitBps_ = fundingSplitBps;
if (fundingSplitRecipient_ != address(0) && fundingSplitBps_ > 0) {
amount = (amount * 1e4) / (1e4 - fundingSplitBps_);
}
}
function finalize() external {
uint96 totalContributions_ = totalContributions;
// Check that the crowdfund is not already finalized.
CrowdfundLifecycle lc = getCrowdfundLifecycle();
if (lc == CrowdfundLifecycle.Active) {
// Allow host to finalize crowdfund early if it has reached its minimum goal.
if (!party.isHost(msg.sender)) revert OnlyPartyHostError();
// Check that the crowdfund has reached its minimum goal.
uint96 minTotalContributions_ = minTotalContributions;
if (totalContributions_ < minTotalContributions_) {
revert NotEnoughContributionsError(totalContributions_, minTotalContributions_);
}
} else {
// Otherwise only allow finalization if the crowdfund has expired
// and been won. Can be finalized by anyone.
if (lc != CrowdfundLifecycle.Won) {
revert WrongLifecycleError(lc);
}
}
// Finalize the crowdfund.
_finalize(totalContributions_);
}
function _finalize(uint96 totalContributions_) internal {
// Finalize the crowdfund.
delete expiry;
// Transfer funding split to recipient if applicable.
address payable fundingSplitRecipient_ = fundingSplitRecipient;
uint16 fundingSplitBps_ = fundingSplitBps;
if (fundingSplitRecipient_ != address(0) && fundingSplitBps_ > 0) {
totalContributions_ -= (totalContributions_ * fundingSplitBps_) / 1e4;
}
// Update the party's total voting power.
uint96 newVotingPower = (totalContributions_ * exchangeRateBps) / 1e4;
party.increaseTotalVotingPower(newVotingPower);
// Transfer ETH to the party.
payable(address(party)).transferEth(totalContributions_);
emit Finalized();
}
/// @notice Send the funding split to the recipient if applicable.
function sendFundingSplit() external returns (uint96 splitAmount) {
// Check that the crowdfund is finalized.
CrowdfundLifecycle lc = getCrowdfundLifecycle();
if (lc != CrowdfundLifecycle.Finalized) revert WrongLifecycleError(lc);
if (fundingSplitPaid) revert FundingSplitAlreadyPaidError();
address payable fundingSplitRecipient_ = fundingSplitRecipient;
uint16 fundingSplitBps_ = fundingSplitBps;
if (fundingSplitRecipient_ == address(0) || fundingSplitBps_ == 0) {
revert FundingSplitNotConfiguredError();
}
fundingSplitPaid = true;
// Transfer funding split to recipient.
splitAmount = (totalContributions * fundingSplitBps_) / 1e4;
payable(fundingSplitRecipient_).transferEth(splitAmount);
emit FundingSplitSent(fundingSplitRecipient_, splitAmount);
}
/// @notice As the DAO, execute an arbitrary function call from this contract.
/// @dev Emergency actions must not be revoked for this to work.
/// @param targetAddress The contract to call.
/// @param targetCallData The data to pass to the contract.
/// @param amountEth The amount of ETH to send to the contract.
function emergencyExecute(
address targetAddress,
bytes calldata targetCallData,
uint256 amountEth
) external payable {
// Must be called by the DAO.
if (_GLOBALS.getAddress(LibGlobals.GLOBAL_DAO_WALLET) != msg.sender) {
revert OnlyPartyDaoError(msg.sender);
}
// Must not be disabled by DAO or host.
if (emergencyExecuteDisabled) {
revert OnlyWhenEmergencyActionsAllowedError();
}
(bool success, bytes memory res) = targetAddress.call{ value: amountEth }(targetCallData);
if (!success) {
res.rawRevert();
}
emit EmergencyExecute(targetAddress, targetCallData, amountEth);
}
/// @notice Revoke the DAO's ability to call emergencyExecute().
/// @dev Either the DAO or the party host can call this.
function disableEmergencyExecute() external {
// Only the DAO or a host can call this.
if (
!party.isHost(msg.sender) &&
_GLOBALS.getAddress(LibGlobals.GLOBAL_DAO_WALLET) != msg.sender
) {
revert OnlyPartyDaoOrHostError(msg.sender);
}
emergencyExecuteDisabled = true;
emit EmergencyExecuteDisabled();
}
}