forked from jtriley-eth/ERC-6909
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathERC6909.vy
125 lines (104 loc) · 4.33 KB
/
ERC6909.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
124
125
# @version 0.3.7
# @title ERC6909 Multi-Token
# @author jtriley.eth
# @notice The event emitted when a transfer occurs.
# @param sender The address of the sender.
# @param receiver The address of the receiver.
# @param id The id of the token.
# @param amount The amount of the token.
event Transfer:
sender: indexed(address)
recipient: indexed(address)
id: indexed(uint256)
amount: uint256
# @notice The event emitted when an operator is set.
# @param owner The address of the owner.
# @param spender The address of the spender.
# @param approved The approval status.
event OperatorSet:
owner: indexed(address)
spender: indexed(address)
approved: bool
# @notice The event emitted when an approval occurs.
# @param owner The address of the owner.
# @param spender The address of the spender.
# @param id The id of the token.
# @param amount The amount of the token.
event Approval:
owner: indexed(address)
spender: indexed(address)
id: indexed(uint256)
amount: uint256
# @notice The total supply of each id.
# @param id The id of the token.
# @return amount The total supply of the token.
totalSupply: public(HashMap[uint256, uint256])
# @notice Owner balance of an id.
# @param owner The address of the owner.
# @param id The id of the token.
# @return amount The balance of the token.
balanceOf: public(HashMap[address, HashMap[uint256, uint256]])
# @notice Spender allowance of an id.
# @param owner The address of the owner.
# @param spender The address of the spender.
# @param id The id of the token.
# @return amount The allowance of the token.
allowance: public(HashMap[address, HashMap[address, HashMap[uint256, uint256]]])
# @notice Checks if a spender is approved by an owner as an operator
# @param owner The address of the owner.
# @param spender The address of the spender.
# @return approved The approval status.
isOperator: public(HashMap[address, HashMap[address, bool]])
# @notice Transfers an amount of an id from the caller to a receiver.
# @param receiver The address of the receiver.
# @param id The id of the token.
# @param amount The amount of the token.
@external
def transfer(receiver: address, id: uint256, amount: uint256) -> bool:
assert self.balanceOf[msg.sender][id] >= amount, "insufficient balance"
self.balanceOf[msg.sender][id] -= amount
self.balanceOf[receiver][id] += amount
log Transfer(msg.sender, receiver, id, amount)
return True
# @notice Transfers an amount of an id from a sender to a receiver.
# @param sender The address of the sender.
# @param receiver The address of the receiver.
# @param id The id of the token.
# @param amount The amount of the token.
@external
def transferFrom(sender: address, receiver: address, id: uint256, amount: uint256) -> bool:
if sender != msg.sender and not self.isOperator[sender][msg.sender]:
sender_allowance: uint256 = self.allowance[sender][msg.sender][id]
assert sender_allowance >= amount, "insufficient allowance"
if sender_allowance != max_value(uint256):
self.allowance[sender][msg.sender][id] = sender_allowance - amount
assert self.balanceOf[sender][id] >= amount, "insufficient balance"
self.balanceOf[sender][id] -= amount
self.balanceOf[receiver][id] += amount
log Transfer(sender, receiver, id, amount)
return True
# @notice Approves an amount of an id to a spender.
# @param spender The address of the spender.
# @param id The id of the token.
# @param amount The amount of the token.
@external
def approve(spender: address, id: uint256, amount: uint256) -> bool:
self.allowance[msg.sender][spender][id] = amount
log Approval(msg.sender, spender, id, amount)
return True
# @notice Sets or removes a spender as an operator for the caller.
# @param spender The address of the spender.
# @param approved The approval status.
@external
def setOperator(spender: address, approved: bool) -> bool:
self.isOperator[msg.sender][spender] = approved
log OperatorSet(msg.sender, spender, approved)
return True
# @notice Checks if a contract implements an interface.
# @param interfaceId The interface identifier, as specified in ERC-165.
# @return supported True if the contract implements `interfaceId` and
# `interfaceId` is not 0xffffffff, false otherwise.
@external
@pure
def supportsInterface(interfaceID: bytes4) -> bool:
return interfaceID in [0x01ffc9a7, 0xb2e69f8a]