-
Notifications
You must be signed in to change notification settings - Fork 0
/
0x595832f8fc6bf59c85c527fec3740a1b7a361269.sol
171 lines (144 loc) · 7.1 KB
/
0x595832f8fc6bf59c85c527fec3740a1b7a361269.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
/**
*Submitted for verification at Etherscan.io on 2017-09-11
*/
pragma solidity 0.4.11;
contract ERC20TokenInterface {
/// @return The total amount of tokens
function totalSupply() constant returns (uint256 supply);
/// @param _owner The address from which the balance will be retrieved
/// @return The balance
function balanceOf(address _owner) constant public returns (uint256 balance);
/// @notice send `_value` token to `_to` from `msg.sender`
/// @param _to The address of the recipient
/// @param _value The amount of token to be transferred
/// @return Whether the transfer was successful or not
function transfer(address _to, uint256 _value) public returns (bool success);
/// @notice send `_value` token to `_to` from `_from` on the condition it is approved by `_from`
/// @param _from The address of the sender
/// @param _to The address of the recipient
/// @param _value The amount of token to be transferred
/// @return Whether the transfer was successful or not
function transferFrom(address _from, address _to, uint256 _value) public returns (bool success);
/// @notice `msg.sender` approves `_spender` to spend `_value` tokens
/// @param _spender The address of the account able to transfer the tokens
/// @param _value The amount of tokens to be approved for transfer
/// @return Whether the approval was successful or not
function approve(address _spender, uint256 _value) public returns (bool success);
/// @param _owner The address of the account owning tokens
/// @param _spender The address of the account able to transfer the tokens
/// @return Amount of remaining tokens allowed to spent
function allowance(address _owner, address _spender) constant public returns (uint256 remaining);
event Transfer(address indexed from, address indexed to, uint256 value);
event Approval(address indexed owner, address indexed spender, uint256 value);
}
contract PowerLedger is ERC20TokenInterface {
//// Constants ////
string public constant name = 'PowerLedger';
uint256 public constant decimals = 6;
string public constant symbol = 'POWR';
string public constant version = '1.0';
string public constant note = 'Democratization of Power';
// One billion coins, each divided to up to 10^decimals units.
uint256 private constant totalTokens = 1000000000 * (10 ** decimals);
mapping (address => uint256) public balances; // (ERC20)
// A mapping from an account owner to a map from approved spender to their allowances.
// (see ERC20 for details about allowances).
mapping (address => mapping (address => uint256)) public allowed; // (ERC20)
//// Events ////
event MigrationInfoSet(string newMigrationInfo);
// This is to be used when migration to a new contract starts.
// This string can be used for any authorative information re the migration
// (e.g. address to use for migration, or URL to explain where to find more info)
string public migrationInfo = "";
// The only address that can set migrationContractAddress, a secure multisig.
address public migrationInfoSetter;
//// Modifiers ////
modifier onlyFromMigrationInfoSetter {
if (msg.sender != migrationInfoSetter) {
throw;
}
_;
}
//// Public functions ////
function PowerLedger(address _migrationInfoSetter) {
if (_migrationInfoSetter == 0) throw;
migrationInfoSetter = _migrationInfoSetter;
// Upon creation, all tokens belong to the deployer.
balances[msg.sender] = totalTokens;
}
// See ERC20
function totalSupply() constant returns (uint256) {
return totalTokens;
}
// See ERC20
// WARNING: If you call this with the address of a contract, the contract will receive the
// funds, but will have no idea where they came from. Furthermore, if the contract is
// not aware of POWR, the tokens will remain locked away in the contract forever.
// It is always recommended to call instead compareAndApprove() (or approve()) and have the
// receiving contract withdraw the money using transferFrom().
function transfer(address _to, uint256 _value) public returns (bool) {
if (balances[msg.sender] >= _value) {
balances[msg.sender] -= _value;
balances[_to] += _value;
Transfer(msg.sender, _to, _value);
return true;
}
return false;
}
// See ERC20
function transferFrom(address _from, address _to, uint256 _value) public returns (bool) {
if (balances[_from] >= _value && allowed[_from][msg.sender] >= _value) {
balances[_from] -= _value;
allowed[_from][msg.sender] -= _value;
balances[_to] += _value;
Transfer(_from, _to, _value);
return true;
}
return false;
}
// See ERC20
function balanceOf(address _owner) constant public returns (uint256) {
return balances[_owner];
}
// See ERC20
// NOTE: this method is vulnerable and is placed here only to follow the ERC20 standard.
// Before using, please take a look at the better compareAndApprove below.
function approve(address _spender, uint256 _value) public returns (bool) {
allowed[msg.sender][_spender] = _value;
Approval(msg.sender, _spender, _value);
return true;
}
// A vulernability of the approve method in the ERC20 standard was identified by
// Mikhail Vladimirov and Dmitry Khovratovich here:
// https://docs.google.com/document/d/1YLPtQxZu1UAvO9cZ1O2RPXBbT0mooh4DYKjA_jp-RLM
// It's better to use this method which is not susceptible to over-withdrawing by the approvee.
/// @param _spender The address to approve
/// @param _currentValue The previous value approved, which can be retrieved with allowance(msg.sender, _spender)
/// @param _newValue The new value to approve, this will replace the _currentValue
/// @return bool Whether the approval was a success (see ERC20's `approve`)
function compareAndApprove(address _spender, uint256 _currentValue, uint256 _newValue) public returns(bool) {
if (allowed[msg.sender][_spender] != _currentValue) {
return false;
}
return approve(_spender, _newValue);
}
// See ERC20
function allowance(address _owner, address _spender) constant public returns (uint256 remaining) {
return allowed[_owner][_spender];
}
// Allows setting a descriptive string, which will aid any users in migrating their token
// to a newer version of the contract. This field provides a kind of 'double-layer' of
// authentication for any migration announcement, as it can only be set by PowerLedger.
/// @param _migrationInfo The information string to be stored on the contract
function setMigrationInfo(string _migrationInfo) onlyFromMigrationInfoSetter public {
migrationInfo = _migrationInfo;
MigrationInfoSet(_migrationInfo);
}
// To be used if the migrationInfoSetter wishes to transfer the migrationInfoSetter
// permission to a new account, e.g. because of change in personnel, a concern that account
// may have been compromised etc.
/// @param _newMigrationInfoSetter The address of the new Migration Info Setter
function changeMigrationInfoSetter(address _newMigrationInfoSetter) onlyFromMigrationInfoSetter public {
migrationInfoSetter = _newMigrationInfoSetter;
}
}