-
Notifications
You must be signed in to change notification settings - Fork 17
/
Copy pathColdStaking.sol
257 lines (211 loc) · 9.1 KB
/
ColdStaking.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
pragma solidity ^0.4.24;
/**
* @title SafeMath
* @dev Math operations with safety checks that throw on error
*/
library SafeMath {
function mul(uint a, uint b) internal pure returns (uint) {
if (a == 0) {
return 0;
}
uint c = a * b;
require(c / a == b);
return c;
}
function div(uint a, uint b) internal pure returns (uint) {
// assert(b > 0); // Solidity automatically throws when dividing by 0
uint c = a / b;
// assert(a == b * c + a % b); // There is no case in which this doesn't hold
return c;
}
function sub(uint a, uint b) internal pure returns (uint) {
require(b <= a);
return a - b;
}
function add(uint a, uint b) internal pure returns (uint) {
uint c = a + b;
require(c >= a);
return c;
}
}
contract ColdStaking {
// NOTE: The contract only works for intervals of time > round_interval
using SafeMath for uint;
event StartStaking(address addr, uint value, uint amount, uint time);
event WithdrawStake(address staker, uint amount);
event Claim(address staker, uint reward);
event DonationDeposited(address _address, uint value);
struct Staker
{
uint amount;
uint time;
}
uint public LastBlock = block.number;
uint public Timestamp = now; //timestamp of the last interaction with the contract.
uint public TotalStakingWeight; //total weight = sum (each_staking_amount * each_staking_time).
uint public TotalStakingAmount; //currently frozen amount for Staking.
uint public StakingRewardPool; //available amount for paying rewards.
bool public CS_frozen; //Cold Staking frozen.
uint public staking_threshold = 0 ether;
address public Treasury = 0x3c06f218ce6dd8e2c535a8925a2edf81674984d9; // Callisto Staking Reserve address.
uint public round_interval = 27 days; // 1 month.
uint public max_delay = 365 * 2 days;// 2 years.
uint public DateStartStaking = 1541980800; // 12.11.2018 0:0:0 UTC.
//========== TESTNET VALUES ===========
//uint public round_interval = 10 minutes;
//uint public max_delay = 2 days;
//uint public DateStartStaking = 0;
//========== END TEST VALUES ==========
mapping(address => Staker) public staker;
function freeze(bool _f) public only_treasurer
{
CS_frozen = _f;
}
function withdraw_rewards () public only_treasurer
{
if (CS_frozen)
{
StakingRewardPool = address(this).balance.sub(TotalStakingAmount);
Treasury.transfer(StakingRewardPool);
}
}
function clear_treasurer () public only_treasurer
{
require(block.number > 1800000 && !CS_frozen);
Treasury = 0x00;
}
function() public payable
{
// No donations accepted to fallback!
// Consider value deposit is an attempt to become staker.
// May not accept deposit from other contracts due GAS limit.
start_staking();
}
// this function can be called for manualy update TotalStakingAmount value.
function new_block() public
{
if (block.number > LastBlock) //run once per block.
{
uint _LastBlock = LastBlock;
LastBlock = block.number;
StakingRewardPool = address(this).balance.sub(TotalStakingAmount + msg.value); //fix rewards pool for this block.
// msg.value here for case new_block() is calling from start_staking(), and msg.value will be added to CurrentBlockDeposits.
//The consensus protocol enforces block timestamps are always atleast +1 from their parent, so a node cannot "lie into the past".
if (now > Timestamp) //But with this condition I feel safer :) May be removed.
{
uint _blocks = block.number - _LastBlock;
uint _seconds = now - Timestamp;
if (_seconds > _blocks * 25) //if time goes far in the future, then use new time as 25 second * blocks.
{
_seconds = _blocks * 25;
}
TotalStakingWeight += _seconds.mul(TotalStakingAmount);
Timestamp += _seconds;
}
}
}
function start_staking() public staking_available payable
{
assert(msg.value >= staking_threshold);
new_block(); //run once per block.
// claim reward if available.
if (staker[msg.sender].amount > 0)
{
if (Timestamp >= staker[msg.sender].time + round_interval)
{
claim();
}
TotalStakingWeight = TotalStakingWeight.sub((Timestamp.sub(staker[msg.sender].time)).mul(staker[msg.sender].amount)); // remove from Weight
}
TotalStakingAmount = TotalStakingAmount.add(msg.value);
staker[msg.sender].time = Timestamp;
staker[msg.sender].amount = staker[msg.sender].amount.add(msg.value);
emit StartStaking(
msg.sender,
msg.value,
staker[msg.sender].amount,
staker[msg.sender].time
);
}
function DEBUG_donation() public payable {
emit DonationDeposited(msg.sender, msg.value);
}
function withdraw_stake() public only_staker
{
new_block(); //run once per block.
require(Timestamp >= staker[msg.sender].time + round_interval); //reject withdrawal before complete round.
uint _amount = staker[msg.sender].amount;
// claim reward if available.
claim();
TotalStakingAmount = TotalStakingAmount.sub(_amount);
TotalStakingWeight = TotalStakingWeight.sub((Timestamp.sub(staker[msg.sender].time)).mul(staker[msg.sender].amount)); // remove from Weight.
staker[msg.sender].amount = 0;
msg.sender.transfer(_amount);
emit WithdrawStake(msg.sender, _amount);
}
//claim rewards
function claim() public only_staker
{
if (CS_frozen) return; //Don't pay rewards when Cold Staking frozen.
new_block(); //run once per block
uint _StakingInterval = Timestamp.sub(staker[msg.sender].time); //time interval of deposit.
if (_StakingInterval >= round_interval)
{
uint _CompleteRoundsInterval = (_StakingInterval / round_interval).mul(round_interval); //only complete rounds.
uint _StakerWeight = _CompleteRoundsInterval.mul(staker[msg.sender].amount); //Weight of completed rounds.
uint _reward = StakingRewardPool.mul(_StakerWeight).div(TotalStakingWeight); //StakingRewardPool * _StakerWeight/TotalStakingWeight
StakingRewardPool = StakingRewardPool.sub(_reward);
TotalStakingWeight = TotalStakingWeight.sub(_StakerWeight); // remove paid Weight.
staker[msg.sender].time = staker[msg.sender].time.add(_CompleteRoundsInterval); // reset to paid time, staking continue without a loss of incomplete rounds.
msg.sender.transfer(_reward);
emit Claim(msg.sender, _reward);
}
}
//This function may be used for info only. This can show estimated user reward at current time.
function stake_reward(address _addr) public constant returns (uint)
{
require(staker[_addr].amount > 0);
require(!CS_frozen);
uint _blocks = block.number - LastBlock;
uint _seconds = now - Timestamp;
if (_seconds > _blocks * 25) //if time goes far in the future, then use new time as 25 second * blocks.
{
_seconds = _blocks * 25;
}
uint _Timestamp = Timestamp + _seconds;
uint _TotalStakingWeight = TotalStakingWeight + _seconds.mul(TotalStakingAmount);
uint _StakingInterval = _Timestamp.sub(staker[_addr].time); //time interval of deposit.
//uint _StakerWeight = _StakingInterval.mul(staker[_addr].amount); //Staker weight.
uint _CompleteRoundsInterval = (_StakingInterval / round_interval).mul(round_interval); //only complete rounds.
uint _StakerWeight = _CompleteRoundsInterval.mul(staker[_addr].amount); //Weight of completed rounds.
uint _StakingRewardPool = address(this).balance.sub(TotalStakingAmount);
return _StakingRewardPool.mul(_StakerWeight).div(_TotalStakingWeight); //StakingRewardPool * _StakerWeight/TotalStakingWeight
}
modifier only_staker
{
require(staker[msg.sender].amount > 0);
_;
}
modifier staking_available
{
require(now >= DateStartStaking && !CS_frozen);
_;
}
modifier only_treasurer
{
require(msg.sender == Treasury);
_;
}
//return deposit to inactive staker.
function report_abuse(address _addr) public only_staker
{
require(staker[_addr].amount > 0);
new_block(); //run once per block.
require(Timestamp > staker[_addr].time.add(max_delay));
uint _amount = staker[_addr].amount;
TotalStakingAmount = TotalStakingAmount.sub(_amount);
TotalStakingWeight = TotalStakingWeight.sub((Timestamp.sub(staker[_addr].time)).mul(_amount)); // remove from Weight.
staker[_addr].amount = 0;
_addr.transfer(_amount);
}
}