-
Notifications
You must be signed in to change notification settings - Fork 4
/
Token.vy
123 lines (100 loc) · 3.77 KB
/
Token.vy
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
# @version ^0.2.0
"""
@title Bare-bones Token implementation
@notice Based on the ERC-20 token standard as defined at
https://eips.ethereum.org/EIPS/eip-20
"""
from vyper.interfaces import ERC20
implements: ERC20
event Approval:
owner: indexed(address)
spender: indexed(address)
value: uint256
event Transfer:
sender: indexed(address)
receiver: indexed(address)
value: uint256
name: public(String[64])
symbol: public(String[32])
decimals: public(uint256)
totalSupply: public(uint256)
balances: HashMap[address, uint256]
allowances: HashMap[address, HashMap[address, uint256]]
@external
def __init__(_name: String[64], _symbol: String[32], _decimals: uint256, _total_supply: uint256):
self.name = _name
self.symbol = _symbol
self.decimals = _decimals
self.balances[msg.sender] = _total_supply
self.totalSupply = _total_supply
log Transfer(ZERO_ADDRESS, msg.sender, _total_supply)
@view
@external
def balanceOf(_owner: address) -> uint256:
"""
@notice Getter to check the current balance of an address
@param _owner Address to query the balance of
@return Token balance
"""
return self.balances[_owner]
@view
@external
def allowance(_owner : address, _spender : address) -> uint256:
"""
@notice Getter to check the amount of tokens that an owner allowed to a spender
@param _owner The address which owns the funds
@param _spender The address which will spend the funds
@return The amount of tokens still available for the spender
"""
return self.allowances[_owner][_spender]
@external
def approve(_spender : address, _value : uint256) -> bool:
"""
@notice Approve an address to spend the specified amount of tokens on behalf of msg.sender
@dev Beware that changing an allowance with this method brings the risk that someone may use both the old
and the new allowance by unfortunate transaction ordering. One possible solution to mitigate this
race condition is to first reduce the spender's allowance to 0 and set the desired value afterwards:
https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
@param _spender The address which will spend the funds.
@param _value The amount of tokens to be spent.
@return Success boolean
"""
self.allowances[msg.sender][_spender] = _value
log Approval(msg.sender, _spender, _value)
return True
@internal
def _transfer(_from: address, _to: address, _value: uint256):
"""
@dev Internal shared logic for transfer and transferFrom
"""
assert self.balances[_from] >= _value, "Insufficient balance"
self.balances[_from] -= _value
self.balances[_to] += _value
log Transfer(_from, _to, _value)
@external
def transfer(_to : address, _value : uint256) -> bool:
"""
@notice Transfer tokens to a specified address
@dev Vyper does not allow underflows, so attempting to transfer more
tokens than an account has will revert
@param _to The address to transfer to
@param _value The amount to be transferred
@return Success boolean
"""
self._transfer(msg.sender, _to, _value)
return True
@external
def transferFrom(_from : address, _to : address, _value : uint256) -> bool:
"""
@notice Transfer tokens from one address to another
@dev Vyper does not allow underflows, so attempting to transfer more
tokens than an account has will revert
@param _from The address which you want to send tokens from
@param _to The address which you want to transfer to
@param _value The amount of tokens to be transferred
@return Success boolean
"""
assert self.allowances[_from][msg.sender] >= _value, "Insufficient allowance"
self.allowances[_from][msg.sender] -= _value
self._transfer(_from, _to, _value)
return True