-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathGHSubManager.sol
713 lines (628 loc) · 22.2 KB
/
GHSubManager.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
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
pragma solidity ^0.4.24;
import "./ISubscription.sol";
import "./Registry.sol";
import "./base/ERC20.sol";
/// @title ERC-948 Enhanced Multi-signature wallet - Allows multiple parties to agree on transactions abd before execution.
/// @author Stefan George - <stefan.george@consensys.net>
/// @author Andrew Redden - <andrew@blockcrushr.com> ERC-948
contract MultiSigWallet {
/*
* Events
*/
event Confirmation(address indexed sender, uint indexed transactionId);
event Revocation(address indexed sender, uint indexed transactionId);
event Submission(uint indexed transactionId);
event Execution(uint indexed transactionId);
event ExecutionFailure(uint indexed transactionId);
event ExecutionSubscription(uint subscriptionId);
event ExecutionSubscriptionFailure(uint subscriptionId);
event Deposit(address indexed sender, uint value);
event OwnerAddition(address indexed owner);
event OwnerRemoval(address indexed owner);
event RequirementChange(uint required);
event AddSubscription(uint id, address indexed txDestination, uint unitAmount, uint period, uint type);
event RevokeSubscription(uint subscriptionId, address indexed destination);
event TCRChanged(address indexed oldTCR, address indexed newTCR);
/*
* views
*/
uint constant public MAX_OWNER_COUNT = 50;
/*
* Storage
*/
mapping(uint => Subscriptions) public subscriptions;
mapping(uint => Transaction) public transactions;
mapping(uint => mapping(address => bool)) public confirmations;
mapping(address => bool) public isOwner;
address[] public owners;
address public tcr;
uint public required;
uint public transactionCount;
uint public subscriptionCount;
struct Transaction {
address destination;
uint value;
bytes data;
bool executed;
}
struct Subscription {
address destination;
address recipient;
address wallet;
uint value;
uint created;
uint expires;
uint cycle;
uint period;
uint lastWithdrawCompleted;
uint withdraw;
uint externalId;
bytes data;
bytes[] meta;
bool pause;
}
/*
* Modifiers
*/
modifier onlyWallet() {
require(msg.sender == address(this));
_;
}
modifier ownerDoesNotExist(address owner) {
require(!isOwner[owner]);
_;
}
modifier ownerExists(address owner) {
require(isOwner[owner]);
_;
}
modifier transactionExists(uint transactionId) {
require(transactions[transactionId].destination != address(0));
_;
}
modifier subscriptionExists(uint subscriptionId) {
require(subscriptions[subscriptionId].destination != address(0));
_;
}
modifier confirmed(uint transactionId, address owner) {
require(confirmations[transactionId][owner]);
_;
}
modifier notConfirmed(uint transactionId, address owner) {
require(!confirmations[transactionId][owner]);
_;
}
modifier notExecuted(uint transactionId) {
require(!transactions[transactionId].executed);
_;
}
modifier subscriptionNotExpired(uint subscriptionId) {
require(!subscriptions[subscriptionId].expire <= now);
_;
}
modifier notNull(address _address) {
require(_address != 0);
_;
}
modifier validRequirement(uint ownerCount, uint _required) {
require(ownerCount <= MAX_OWNER_COUNT
&& _required <= ownerCount
&& _required != 0
&& ownerCount != 0);
_;
}
/// @dev Fallback function allows to deposit ether.
function()
payable
{
if (msg.value > 0)
emit Deposit(msg.sender, msg.value);
}
/*
* Public functions
*/
/// @dev Contract constructor sets initial owners and required number of confirmations.
/// @param _owners List of initial owners.
/// @param _required Number of required confirmations.
constructor (address[] _owners, uint _required, address _tcr)
payable
public
validRequirement(_owners.length, _required)
{
for (uint i = 0; i < _owners.length; i++) {
require(!isOwner[_owners[i]] && _owners[i] != 0);
isOwner[_owners[i]] = true;
}
owners = _owners;
required = _required;
tcr = _tcr;
if (msg.value > 0) {
emit Deposit(msg.sender, msg.value);
}
}
/// @dev Allows to add a new owner. Transaction has to be sent by wallet.
/// @param owner Address of new owner.
function addOwner(address _owner)
public
onlyWallet
ownerDoesNotExist(_owner)
notNull(_owner)
validRequirement(owners.length + 1, required)
{
isOwner[_owner] = true;
owners.push(_owner);
emit OwnerAddition(_owner);
}
/// @dev Allows to remove an owner. Transaction has to be sent by wallet.
/// @param owner Address of owner.
function removeOwner(address _owner)
public
onlyWallet
ownerExists(_owner)
{
isOwner[_owner] = false;
for (uint i = 0; i < owners.length - 1; i++)
if (owners[i] == _owner) {
owners[i] = owners[owners.length - 1];
break;
}
owners.length -= 1;
if (required > owners.length)
changeRequirement(owners.length);
emit OwnerRemoval(_owner);
}
/// @dev Allows to replace an owner with a new owner. Transaction has to be sent by wallet.
/// @param owner Address of owner to be replaced.
/// @param newOwner Address of new owner.
function replaceOwner(address _owner, address _newOwner)
public
onlyWallet
ownerExists(_owner)
ownerDoesNotExist(_newOwner)
{
for (uint i = 0; i < owners.length; i++)
if (owners[i] == _owner) {
owners[i] = _newOwner;
break;
}
isOwner[_owner] = false;
isOwner[_newOwner] = true;
emit OwnerRemoval(_owner);
emit OwnerAddition(_newOwner);
}
/// @dev Allows to change the number of required confirmations. Transaction has to be sent by wallet.
/// @param _required Number of required confirmations.
function changeRequirement(uint _required)
public
onlyWallet
validRequirement(owners.length, _required)
{
required = _required;
emit RequirementChange(_required);
}
function changeTCRAddress(address _newTCR)
public
ownerExists(msg.sender) {
tcr = _newTCR;
emit TCRChanged(_newTCR);
}
/// @dev Allows an owner to submit and confirm a transaction.
/// @param destination Transaction target address.
/// @param value Transaction ether value.
/// @param data Transaction data payload.
/// @return Returns transaction ID.
function submitTransaction(address tx_destination, uint _value, bytes _data)
public
returns (uint transactionId)
{
transactionId = addTransaction(tx_destination, _value, _data);
confirmTransaction(transactionId);
}
/// @dev Allows an owner to submit and confirm a subscription.
/// @param destination Subscription target address.
/// @param recipient Subscription recipient address (optional)
/// @param value Subscription value, depends on type.
/// @param type Subscription type
/// @param period Subscription withdrawal period
/// @param data Subscription data payload.
/// @param meta Subscription meta data payload.
/// @return Returns subscriptionId.
function submitSubscription(
address _txDestination,
address _recipient,
uint _value,
uint _period,
uint _type,
bytes _data,
bytes[] _meta
)
payable
public
returns (uint subscriptionId)
{
subscriptionId = addSubscription(_txDestination, _recipient, _type, _value, _period, _data, _meta);
if (address(tcr) != address(0)) {
ITCR dest = ITCR(tcr);
dest.handleNewSubscription(_txDestination, address(this), subscriptionId, _externalId);
}
if (msg.value >= _value || (this.balance >= _value) || _data != null) {// check to see if payment is valid
executeSubscription(subscriptionId);
}
emit NewSubscription(subscriptionId, _txDestination, _value, _period, _type);
}
/// @dev Allows an owner to submit and confirm a transaction.
/// @param destination Transaction target address.
/// @param value Transaction ether value.
/// @param data Transaction data payload.
/// @param expires Transaction data payload.
/// @param period Transaction data payload.
/// @param externalId Transaction data payload.
/// @return Returns transaction ID.
function cancelSubscription(uint _subscriptionId)
public
ownerExists(msg.sender)
returns (uint subscriptionId)
{
//load subscription into storage, modify expire to match today
Subscription storage sub = subscriptions[_subscriptionId];
sub.expires = now;
emit CancelSubscription(address(this), _subscriptionId);
}
/// @dev Allows an owner to confirm a transaction.
/// @param transactionId Transaction ID.
function confirmTransaction(uint transactionId)
public
ownerExists(msg.sender)
transactionExists(transactionId)
notConfirmed(transactionId, msg.sender)
{
confirmations[transactionId][msg.sender] = true;
emit Confirmation(msg.sender, transactionId);
executeTransaction(transactionId);
}
/// @dev Allows an owner to revoke a confirmation for a transaction.
/// @param transactionId Transaction ID.
function revokeConfirmation(uint transactionId)
public
ownerExists(msg.sender)
confirmed(transactionId, msg.sender)
notExecuted(transactionId)
{
confirmations[transactionId][msg.sender] = false;
emit Revocation(msg.sender, transactionId);
}
modifier validOperator(address _operator) {
if (address(tcr) != address(0)) {
ITCR itcr = ITCR(tcr);
require(repo.isOperator[_operator]);
} else {
require(isOwner[owner]);
}
_;
}
modifier validWithdrawal(uint subscriptionId) {
require(subscriptions[subscriptionId].expires > now);
require(subscriptions[subscriptionId].nextWithdraw <= now);
require(address(this).balance >= subscriptions[subscriptionId].value);
_;
}
/// @dev Allows valid operators to execute a valid subscription
/// @param subscriptionId Subscription ID.
function executeSubscription(uint subscriptionId)
public
validOperator(msg.sender)
validWithdrawal(subscriptionId)
{
//check type of subscription, if normal txn, build and do external call
Subscription storage sub = subscriptions[subscriptionId];
bool success = false;
if (sub.type == uint(0)) {
if (external_call(sub.destination, sub.value, sub.data.length, sub.data)) {
success = true;
}
} else if (sub.type == uint(1)) {
if (transferFrom(sub.txDestination, sub.wallet, sub.recipient, sub.value)) {
success = true;
}
} else {
revert("Un implemented type");
}
if (success) {
Subscription storage sub = subscriptions[subscriptionId];
sub.lastWithdrawalCompleted = now;
sub.nextWithdraw = sub.created + (sub.period * sub.cycle);
emit ExecutionSubscription(subscriptionId);
if (address(tcr) != address(0)) {
ITCR dest = ITCR(tcr);
bool first = false;
if (sub.cycle == 0) {
first = true;
}
dest.handlePaymentNotification(tx_destination, subscriptionId, sub.externalId, first);
}
sub.cycle++;
} else {
emit ExecutionSubscriptionFailure(subscriptionId);
}
}
/// @dev Allows anyone to execute a confirmed transaction.
/// @param transactionId Transaction ID.
function executeTransaction(uint transactionId)
public
ownerExists(msg.sender)
confirmed(transactionId, msg.sender)
notExecuted(transactionId)
{
if (isConfirmed(transactionId)) {
Transaction storage txn = transactions[transactionId];
txn.executed = true;
if (external_call(txn.destination, txn.value, txn.data.length, txn.data))
emit Execution(transactionId);
else {
emit ExecutionFailure(transactionId);
txn.executed = false;
}
}
}
// call has been separated into its own function in order to take advantage
// of the Solidity's code generator to produce a loop that copies tx.data into memory.
function external_call(address destination, uint value, uint dataLength, bytes data) private returns (bool) {
bool result;
assembly {
let x := mload(0x40) // "Allocate" memory for output (0x40 is where "free memory" pointer is stored by convention)
let d := add(data, 32) // First 32 bytes are the padded length of data, so exclude that
result := call(
sub(gas, 34710), // 34710 is the value that solidity is currently emitting
// It includes callGas (700) + callVeryLow (3, to pay for SUB) + callValueTransferGas (9000) +
// callNewAccountGas (25000, in case the destination address does not exist and needs creating)
destination,
value,
d,
dataLength, // Size of the input (in bytes) - this is what fixes the padding problem
x,
0 // Output is ignored, therefore the output size is zero
)
}
return result;
}
/// @dev Returns the confirmation status of a transaction.
/// @param transactionId Transaction ID.
/// @return Confirmation status.
function isConfirmed(uint transactionId)
public
view
returns (bool)
{
uint count = 0;
for (uint i = 0; i < owners.length; i++) {
if (confirmations[transactionId][owners[i]])
count += 1;
if (count == required)
return true;
}
}
/*
* Internal functions
*/
/// @dev Adds a new transaction to the transaction mapping, if transaction does not exist yet.
/// @param destination Transaction target address.
/// @param value Transaction ether value.
/// @param data Transaction data payload.
/// @return Returns transaction ID.
function addTransaction(address destination, uint value, bytes data)
internal
notNull(destination)
returns (uint transactionId)
{
transactionId = transactionCount;
transactions[transactionId] = Transaction({
destination : destination,
value : value,
data : data,
executed : false
});
transactionCount += 1;
emit Submission(transactionId);
}
/*
* Internal functions
*/
/// @dev Adds a new transaction to the transaction mapping, if transaction does not exist yet.
/// @param _txDestination Subscription target address, smart contract, normal wallet, etc.
/// @param _txDestination Subscription target recipient address, smart contract, normal wallet, etc.
/// @param _value Transaction ether or token value.
/// @param _type Enumerated tx type
/// @param _period Transaction data payload.
/// @param _data Transaction data payload.
/// @param _meta Transaction data payload.
/// @param _data Transaction data payload.
/// @return Returns transaction ID.
function addSubscription(
address _txDestination,
address _recipient,
uint _value,
uint _type,
uint _period,
bytes _data,
bytes[] _meta)
internal
notNull(destination)
validType(_type)
returns (uint subscriptionId)
{
//hash the destination and the data to make sure we don't have a copy of the subscription
//check meta array for expires, externalId
//check type value, then check meta data to match the schema, wallet
address wallet = address(this);
bytes externalId = "";
uint expires = (2 * 256) - 1;
subscriptionId = subscriptionCount;
subscriptions[subscriptionId] = Subscription({
destination : _txDestination,
recipient : _recipient,
wallet : wallet,
value : value,
data : data,
created : now,
expires : expires,
period : period,
externalId : externalId
});
subscriptionCount += 1;
emit AddSubscription(subscriptionId);
}
/*
* Web3 call functions
*/
/// @dev Returns number of confirmations of a transaction.
/// @param transactionId Transaction ID.
/// @return Number of confirmations.
function getConfirmationCount(uint transactionId)
public
view
returns (uint count)
{
for (uint i = 0; i < owners.length; i++)
if (confirmations[transactionId][owners[i]])
count += 1;
}
/// @dev Returns total number of transactions after filers are applied.
/// @param pending Include pending transactions.
/// @param executed Include executed transactions.
/// @return Total number of transactions after filters are applied.
function getTransactionCount(bool pending, bool executed)
public
view
returns (uint count)
{
for (uint i = 0; i < transactionCount; i++)
if (pending && !transactions[i].executed
|| executed && transactions[i].executed)
count += 1;
}
/// @dev Returns total number of transactions after filers are applied.
/// @param withdraw Include transactions available for withdraw.
/// @param expired Include expired subscriptions.
/// @return Total number of transactions after filters are applied.
function getSubscriptionCount(bool withdraw, bool expired)
public
view
returns (uint count)
{
for (uint i = 0; i < subscriptionCount; i++) {
if (withdraw) {
if (subscriptions[i].withdraw <= now) {
subscriptionIdsTemp[count] = i;
count += 1;
}
} else if (expired) {
if (subscriptions[i].expired >= now) {
subscriptionIdsTemp[count] = i;
count += 1;
}
} else {
subscriptionIdsTemp[count] = i;
count += 1;
}
}
}
/// @dev Returns list of owners.
/// @return List of owner addresses.
function getOwners()
public
view
returns (address[])
{
return owners;
}
/// @dev Returns array with owner addresses, which confirmed transaction.
/// @param transactionId Transaction ID.
/// @return Returns array of owner addresses.
function getConfirmations(uint transactionId)
public
view
returns (address[] _confirmations)
{
address[] memory confirmationsTemp = new address[](owners.length);
uint count = 0;
uint i;
for (i = 0; i < owners.length; i++)
if (confirmations[transactionId][owners[i]]) {
confirmationsTemp[count] = owners[i];
count += 1;
}
_confirmations = new address[](count);
for (i = 0; i < count; i++)
_confirmations[i] = confirmationsTemp[i];
}
/// @dev Returns list of transaction IDs in defined range.
/// @param from Index start position of transaction array.
/// @param to Index end position of transaction array.
/// @param pending Include pending transactions.
/// @param executed Include executed transactions.
/// @return Returns array of transaction IDs.
function getTransactionIds(uint from, uint to, bool pending, bool executed)
public
view
returns (uint[] _transactionIds)
{
uint[] memory transactionIdsTemp = new uint[](transactionCount);
uint count = 0;
uint i;
for (i = 0; i < transactionCount; i++)
if (pending && !transactions[i].executed
|| executed && transactions[i].executed)
{
transactionIdsTemp[count] = i;
count += 1;
}
_transactionIds = new uint[](to - from);
for (i = from; i < to; i++)
_transactionIds[i - from] = transactionIdsTemp[i];
}
/// @dev Returns list of subscription IDs in defined range.
/// @param from Index start position of transaction array.
/// @param to Index end position of transaction array.
/// @param withdraw Include subscriptions that can be withdrawn.
/// @param expired include expired subscriptions
/// @return Returns array of subscription IDs.
function getSubscriptionIds(uint from, uint to, bool withdraw, bool expired)
public
view
returns (uint[] _subscriptionIds)
{
uint[] memory subscriptionIdsTemp = new uint[](subscriptionCount);
uint count = 0;
uint i;
for (i = 0; i < subscriptionCount; i++)
if (withdraw) {
if (subscriptions[i].nextWithdraw <= now) {
subscriptionIdsTemp[count] = i;
count += 1;
}
} else if (expired) {
if (subscriptions[i].expired >= now) {
subscriptionIdsTemp[count] = i;
count += 1;
}
} else {
subscriptionIdsTemp[count] = i;
count += 1;
}
_subscriptionIds = new uint[](to - from);
for (i = from; i < to; i++) {
_subscriptionIds[i - from] = subscriptionIdsTemp[i];
}
}
function transferFrom(
address _token,
address _from,
address _to,
uint _value)
internal
validOperator(msg.sender)
returns (bool success)
{
return ERC20(_token).transferFrom(_from, _to, _value);
}
}