@leofcoin/chain 1.4.5 → 1.4.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +2 -2
- package/src/contract.js +4 -3
- package/src/machine.js +22 -10
- package/src/contracts/factory.js +0 -52
- package/src/contracts/name-service.js +0 -106
- package/src/contracts/native-token.js +0 -7
- package/src/contracts/power-token.js +0 -7
- package/src/contracts/validators.js +0 -129
- package/src/standards/roles.js +0 -65
- package/src/standards/token.js +0 -137
- package/src/standards/voting.js +0 -1
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@leofcoin/chain",
|
|
3
|
-
"version": "1.4.
|
|
3
|
+
"version": "1.4.7",
|
|
4
4
|
"description": "Official javascript implementation",
|
|
5
5
|
"main": "./dist/node.js",
|
|
6
6
|
"module": "./dist/chain.esm",
|
|
@@ -18,7 +18,7 @@
|
|
|
18
18
|
"workers": "cp ./../workers/src/** ./workers",
|
|
19
19
|
"build": "npm run c && webpack && cp ./dist/*browser.js ./demo && cp -r ./dist/browser/workers/ ./demo",
|
|
20
20
|
"demo": "jsproject --serve ./demo --port 5478 --open demo",
|
|
21
|
-
"test": "node
|
|
21
|
+
"test": "node --openssl-legacy-provider test",
|
|
22
22
|
"pack": "webpack"
|
|
23
23
|
},
|
|
24
24
|
"np": {
|
package/src/contract.js
CHANGED
|
@@ -1,5 +1,6 @@
|
|
|
1
1
|
import Transaction from "./transaction.js";
|
|
2
2
|
import { createContractMessage } from '@leofcoin/lib'
|
|
3
|
+
import addresses from "@leofcoin/addresses"
|
|
3
4
|
|
|
4
5
|
/**
|
|
5
6
|
* @extends {Transaction}
|
|
@@ -29,7 +30,7 @@ export default class Contract extends Transaction {
|
|
|
29
30
|
*/
|
|
30
31
|
async createContractAddress(creator, contract, constructorParameters = []) {
|
|
31
32
|
contract = await this.createContractMessage(creator, contract, constructorParameters)
|
|
32
|
-
return contract.hash
|
|
33
|
+
return contract.hash()
|
|
33
34
|
}
|
|
34
35
|
|
|
35
36
|
/**
|
|
@@ -41,11 +42,11 @@ export default class Contract extends Transaction {
|
|
|
41
42
|
async deployContract(contract, constructorParameters = []) {
|
|
42
43
|
const message = await createContractMessage(peernet.selectedAccount, contract, constructorParameters)
|
|
43
44
|
try {
|
|
44
|
-
await contractStore.put(await message.hash, message.encoded)
|
|
45
|
+
await contractStore.put(await message.hash(), message.encoded)
|
|
45
46
|
} catch (error) {
|
|
46
47
|
throw error
|
|
47
48
|
}
|
|
48
|
-
return this.createTransactionFrom(peernet.selectedAccount, addresses.contractFactory, 'registerContract', [await message.hash])
|
|
49
|
+
return this.createTransactionFrom(peernet.selectedAccount, addresses.contractFactory, 'registerContract', [await message.hash()])
|
|
49
50
|
}
|
|
50
51
|
|
|
51
52
|
}
|
package/src/machine.js
CHANGED
|
@@ -122,7 +122,7 @@ export default class Machine {
|
|
|
122
122
|
async execute(contract, method, parameters) {
|
|
123
123
|
try {
|
|
124
124
|
if (contract === contractFactory && method === 'registerContract') {
|
|
125
|
-
if (this
|
|
125
|
+
if (await this.has(parameters[0])) throw new Error(`duplicate contract @${parameters[0]}`)
|
|
126
126
|
let message;
|
|
127
127
|
if (!await contractStore.has(parameters[0])) {
|
|
128
128
|
message = await peernet.get(parameters[0], 'contract')
|
|
@@ -133,7 +133,7 @@ export default class Machine {
|
|
|
133
133
|
message = await contractStore.get(parameters[0])
|
|
134
134
|
message = await new ContractMessage(message)
|
|
135
135
|
}
|
|
136
|
-
if (!this
|
|
136
|
+
if (!await this.has(await message.hash())) await this.#runContract(message)
|
|
137
137
|
}
|
|
138
138
|
} catch (error) {
|
|
139
139
|
throw new Error(`contract deployment failed for ${parameters[0]}\n${error.message}`)
|
|
@@ -159,14 +159,6 @@ export default class Machine {
|
|
|
159
159
|
|
|
160
160
|
}
|
|
161
161
|
|
|
162
|
-
addJob(contract, method, parameters, from, nonce) {
|
|
163
|
-
if (!this.#nonces[from]) this.#nonces[from] = nonce
|
|
164
|
-
if (nonce === this.#nonces[from] + 1) return this.#contracts[contract][method](...parameters)
|
|
165
|
-
// return setTimeout(() => {
|
|
166
|
-
// return this.addJob(contract, method, params, from, nonce)
|
|
167
|
-
// }, 50)
|
|
168
|
-
}
|
|
169
|
-
|
|
170
162
|
get(contract, method, parameters) {
|
|
171
163
|
return new Promise((resolve, reject) => {
|
|
172
164
|
const id = randombytes(20).toString()
|
|
@@ -187,6 +179,26 @@ export default class Machine {
|
|
|
187
179
|
})
|
|
188
180
|
}
|
|
189
181
|
|
|
182
|
+
|
|
183
|
+
async has(address) {
|
|
184
|
+
return new Promise((resolve, reject) => {
|
|
185
|
+
const id = randombytes(20).toString('hex')
|
|
186
|
+
const onmessage = message => {
|
|
187
|
+
pubsub.unsubscribe(id, onmessage)
|
|
188
|
+
if (message?.error) reject(message.error)
|
|
189
|
+
else resolve(message)
|
|
190
|
+
}
|
|
191
|
+
pubsub.subscribe(id, onmessage)
|
|
192
|
+
this.worker.postMessage({
|
|
193
|
+
type: 'has',
|
|
194
|
+
id,
|
|
195
|
+
input: {
|
|
196
|
+
address
|
|
197
|
+
}
|
|
198
|
+
})
|
|
199
|
+
})
|
|
200
|
+
}
|
|
201
|
+
|
|
190
202
|
async delete(hash) {
|
|
191
203
|
return contractStore.delete(hash)
|
|
192
204
|
}
|
package/src/contracts/factory.js
DELETED
|
@@ -1,52 +0,0 @@
|
|
|
1
|
-
export default class Factory {
|
|
2
|
-
/**
|
|
3
|
-
* string
|
|
4
|
-
*/
|
|
5
|
-
#name = 'ArtOnlineContractFactory'
|
|
6
|
-
/**
|
|
7
|
-
* uint
|
|
8
|
-
*/
|
|
9
|
-
#totalContracts = 0
|
|
10
|
-
/**
|
|
11
|
-
* Array => string
|
|
12
|
-
*/
|
|
13
|
-
#contracts = []
|
|
14
|
-
|
|
15
|
-
constructor(state) {
|
|
16
|
-
if (state) {
|
|
17
|
-
this.#contracts = state.contracts
|
|
18
|
-
this.#totalContracts = state.totalContracts
|
|
19
|
-
}
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
get state() {
|
|
23
|
-
return {
|
|
24
|
-
totalContracts: this.#totalContracts,
|
|
25
|
-
contracts: this.#contracts
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
get name() {
|
|
30
|
-
return this.#name
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
get contracts() {
|
|
34
|
-
return [...this.#contracts]
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
get totalContracts() {
|
|
38
|
-
return this.#totalContracts
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
/**
|
|
42
|
-
*
|
|
43
|
-
* @param {Address} address contract address to register
|
|
44
|
-
*/
|
|
45
|
-
async registerContract(address) {
|
|
46
|
-
await msg.staticCall(address, 'hasRole', [msg.sender, 'OWNER'])
|
|
47
|
-
if (this.#contracts.includes(address)) throw new Error('already registered')
|
|
48
|
-
|
|
49
|
-
this.#totalContracts += 1
|
|
50
|
-
this.#contracts.push(address)
|
|
51
|
-
}
|
|
52
|
-
}
|
|
@@ -1,106 +0,0 @@
|
|
|
1
|
-
export default class NameService {
|
|
2
|
-
/**
|
|
3
|
-
* string
|
|
4
|
-
*/
|
|
5
|
-
#name = 'ArtOnlineNameService'
|
|
6
|
-
/**
|
|
7
|
-
* string
|
|
8
|
-
*/
|
|
9
|
-
#owner
|
|
10
|
-
/**
|
|
11
|
-
* number
|
|
12
|
-
*/
|
|
13
|
-
#price = 0
|
|
14
|
-
/**
|
|
15
|
-
* Object => string
|
|
16
|
-
*/
|
|
17
|
-
#registry = {}
|
|
18
|
-
|
|
19
|
-
/**
|
|
20
|
-
* => string
|
|
21
|
-
*/
|
|
22
|
-
#currency
|
|
23
|
-
|
|
24
|
-
get name() {
|
|
25
|
-
return this.#name
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
get registry() {
|
|
29
|
-
return {...this.#registry}
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
get state() {}
|
|
33
|
-
|
|
34
|
-
// TODO: control with contract
|
|
35
|
-
constructor(factoryAddress, currency, validatorAddress, price, state) {
|
|
36
|
-
if (state) {
|
|
37
|
-
this.#owner = state.owner
|
|
38
|
-
this.#registry = state.registry
|
|
39
|
-
this.#currency = state.currency
|
|
40
|
-
this.#price = state.price
|
|
41
|
-
} else {
|
|
42
|
-
this.#owner = msg.sender
|
|
43
|
-
this.#price = price
|
|
44
|
-
this.#registry['ArtOnlineContractFactory'] = {
|
|
45
|
-
owner: msg.sender,
|
|
46
|
-
address: factoryAddress
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
this.#registry['ArtOnlineToken'] = {
|
|
50
|
-
owner: msg.sender,
|
|
51
|
-
address: currency
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
this.#registry['ArtOnlineValidators'] = {
|
|
55
|
-
owner: msg.sender,
|
|
56
|
-
address: validatorAddress
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
this.#currency = currency
|
|
60
|
-
}
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
changeOwner(owner) {
|
|
64
|
-
if (msg.sender !== this.#owner) throw new Error('no owner')
|
|
65
|
-
this.#owner = owner
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
changePrice(price) {
|
|
69
|
-
if (msg.sender !== this.#owner) throw new Error('no owner')
|
|
70
|
-
this.#price = price
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
changeCurrency(currency) {
|
|
74
|
-
if (msg.sender !== this.#owner) throw new Error('no owner')
|
|
75
|
-
this.#currency = currency
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
async purchaseName(name, address) {
|
|
79
|
-
const balance = await msg.call(this.#currency, 'balanceOf', [msg.sender])
|
|
80
|
-
if (balance < this.#price) throw new Error('price exceeds balance')
|
|
81
|
-
try {
|
|
82
|
-
await msg.call(this.#currency, 'transfer', [msg.sender, this.#owner, this.#price])
|
|
83
|
-
} catch (error) {
|
|
84
|
-
throw error
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
this.#registry[name] = {
|
|
88
|
-
owner: msg.sender,
|
|
89
|
-
address
|
|
90
|
-
}
|
|
91
|
-
}
|
|
92
|
-
|
|
93
|
-
lookup(name) {
|
|
94
|
-
return this.#registry[name]
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
transferOwnership(name, to) {
|
|
98
|
-
if (msg.sender !== this.#registry.owner) throw new Error('not a owner')
|
|
99
|
-
this.#registry[name].owner = to
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
changeAddress(name, address) {
|
|
103
|
-
if (msg.sender !== this.#registry.owner) throw new Error('not a owner')
|
|
104
|
-
this.#registry[name].address = address
|
|
105
|
-
}
|
|
106
|
-
}
|
|
@@ -1,129 +0,0 @@
|
|
|
1
|
-
import Roles from './../standards/roles'
|
|
2
|
-
|
|
3
|
-
export default class Validators extends Roles {
|
|
4
|
-
/**
|
|
5
|
-
* string
|
|
6
|
-
*/
|
|
7
|
-
#name = 'ArtOnlineValidators'
|
|
8
|
-
/**
|
|
9
|
-
* uint
|
|
10
|
-
*/
|
|
11
|
-
#totalValidators = 0
|
|
12
|
-
|
|
13
|
-
#activeValidators = 0
|
|
14
|
-
/**
|
|
15
|
-
* Object => string(address) => Object
|
|
16
|
-
*/
|
|
17
|
-
#validators = {}
|
|
18
|
-
|
|
19
|
-
#currency
|
|
20
|
-
|
|
21
|
-
#minimumBalance
|
|
22
|
-
|
|
23
|
-
get state() {
|
|
24
|
-
return {
|
|
25
|
-
...super.state,
|
|
26
|
-
minimumBalance: this.#minimumBalance,
|
|
27
|
-
currency: this.#currency,
|
|
28
|
-
totalValidators: this.#totalValidators,
|
|
29
|
-
activeValidators: this.#activeValidators,
|
|
30
|
-
validators: this.#validators
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
|
|
34
|
-
constructor(tokenAddress, state) {
|
|
35
|
-
super(state?.roles)
|
|
36
|
-
if (state) {
|
|
37
|
-
this.#minimumBalance = state.minimumBalance
|
|
38
|
-
this.#currency = state.currency
|
|
39
|
-
|
|
40
|
-
this.#totalValidators = state.totalValidators
|
|
41
|
-
this.#activeValidators = state.activeValidators
|
|
42
|
-
this.#validators = state.validators
|
|
43
|
-
} else {
|
|
44
|
-
this.#minimumBalance = 50_000
|
|
45
|
-
this.#currency = tokenAddress
|
|
46
|
-
|
|
47
|
-
this.#totalValidators += 1
|
|
48
|
-
this.#activeValidators += 1
|
|
49
|
-
this.#validators[msg.sender] = {
|
|
50
|
-
firstSeen: Date.now(),
|
|
51
|
-
lastSeen: Date.now(),
|
|
52
|
-
active: true
|
|
53
|
-
}
|
|
54
|
-
}
|
|
55
|
-
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
get name() {
|
|
59
|
-
return this.#name
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
get currency() {
|
|
63
|
-
return this.#currency
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
get validators() {
|
|
67
|
-
return {...this.#validators}
|
|
68
|
-
}
|
|
69
|
-
|
|
70
|
-
get totalValidators() {
|
|
71
|
-
return this.#totalValidators
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
get minimumBalance() {
|
|
75
|
-
return this.#minimumBalance
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
changeCurrency(currency) {
|
|
79
|
-
if (!this.hasRole(msg.sender, 'OWNER')) throw new Error('not an owner')
|
|
80
|
-
this.#currency = currency
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
has(validator) {
|
|
84
|
-
return Boolean(this.#validators[validator] !== undefined)
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
#isAllowed(address) {
|
|
88
|
-
if (msg.sender !== address && !this.hasRole(msg.sender, 'OWNER')) throw new Error('sender is not the validator or owner')
|
|
89
|
-
return true
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
async addValidator(validator) {
|
|
93
|
-
this.#isAllowed(validator)
|
|
94
|
-
if (this.has(validator)) throw new Error('already a validator')
|
|
95
|
-
|
|
96
|
-
const balance = await msg.staticCall(this.currency, 'balanceOf', [validator])
|
|
97
|
-
|
|
98
|
-
if (balance < this.minimumBalance) throw new Error(`balance to low! got: ${balance} need: ${this.#minimumBalance}`)
|
|
99
|
-
|
|
100
|
-
this.#totalValidators += 1
|
|
101
|
-
this.#activeValidators += 1
|
|
102
|
-
this.#validators[validator] = {
|
|
103
|
-
firstSeen: Date.now(),
|
|
104
|
-
lastSeen: Date.now(),
|
|
105
|
-
active: true
|
|
106
|
-
}
|
|
107
|
-
}
|
|
108
|
-
|
|
109
|
-
removeValidator(validator) {
|
|
110
|
-
this.#isAllowed(validator)
|
|
111
|
-
if (!this.has(validator)) throw new Error('validator not found')
|
|
112
|
-
|
|
113
|
-
this.#totalValidators -= 1
|
|
114
|
-
if (this.#validators[validator].active) this.#activeValidators -= 1
|
|
115
|
-
delete this.#validators[validator]
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
async updateValidator(validator, active) {
|
|
119
|
-
this.#isAllowed(validator)
|
|
120
|
-
if (!this.has(validator)) throw new Error('validator not found')
|
|
121
|
-
const balance = await msg.staticCall(this.currency, 'balanceOf', [validator])
|
|
122
|
-
if (balance < this.minimumBalance && active) throw new Error(`balance to low! got: ${balance} need: ${this.#minimumBalance}`)
|
|
123
|
-
if (this.#validators[validator].active === active) throw new Error(`already ${active ? 'activated' : 'deactivated'}`)
|
|
124
|
-
if (active) this.#activeValidators += 1
|
|
125
|
-
else this.#activeValidators -= 1
|
|
126
|
-
/** minimum balance always needs to be met */
|
|
127
|
-
this.#validators[validator].active = active
|
|
128
|
-
}
|
|
129
|
-
}
|
package/src/standards/roles.js
DELETED
|
@@ -1,65 +0,0 @@
|
|
|
1
|
-
export default class Roles {
|
|
2
|
-
|
|
3
|
-
/**
|
|
4
|
-
* Object => Array
|
|
5
|
-
*/
|
|
6
|
-
#roles = {
|
|
7
|
-
'OWNER': [],
|
|
8
|
-
'MINT': [],
|
|
9
|
-
'BURN': []
|
|
10
|
-
}
|
|
11
|
-
|
|
12
|
-
constructor(roles) {
|
|
13
|
-
// allow devs to set their own roles but always keep the default ones included
|
|
14
|
-
// also allows roles to be loaded from the stateStore
|
|
15
|
-
// carefull when including the roles make sure to add the owner
|
|
16
|
-
// because no roles are granted by default when using custom roles
|
|
17
|
-
if (roles) {
|
|
18
|
-
if (roles instanceof Object) {
|
|
19
|
-
this.#roles = {...roles, ...this.#roles}
|
|
20
|
-
} else {
|
|
21
|
-
throw new TypeError(`expected roles to be an object`)
|
|
22
|
-
}
|
|
23
|
-
} else {
|
|
24
|
-
// no roles given so fallback to default to the msg sender
|
|
25
|
-
this.#grantRole(msg.sender, 'OWNER')
|
|
26
|
-
}
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
get state() {
|
|
30
|
-
return { roles: this.roles }
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
get roles() {
|
|
34
|
-
return {...this.#roles}
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
hasRole(address, role) {
|
|
38
|
-
return this.#roles[role] ? this.#roles[role].includes(address) : false
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
#grantRole(address, role) {
|
|
42
|
-
if (this.hasRole(address, role)) throw new Error(`${role} role already granted for ${address}`)
|
|
43
|
-
|
|
44
|
-
this.#roles[role].push(address)
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
#revokeRole(address, role) {
|
|
48
|
-
if (!this.hasRole(address, role)) throw new Error(`${role} role already revoked for ${address}`)
|
|
49
|
-
if (role === 'OWNER' && this.#roles[role].length === 1) throw new Error(`atleast one owner is needed!`)
|
|
50
|
-
|
|
51
|
-
this.#roles[role].splice(this.#roles[role].indexOf(address))
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
grantRole(address, role) {
|
|
55
|
-
if (!this.hasRole(address, 'OWNER')) throw new Error('Not allowed')
|
|
56
|
-
|
|
57
|
-
this.#grantRole(address, role)
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
revokeRole(address, role) {
|
|
61
|
-
if (!this.hasRole(address, 'OWNER')) throw new Error('Not allowed')
|
|
62
|
-
|
|
63
|
-
this.#revokeRole(address, role)
|
|
64
|
-
}
|
|
65
|
-
}
|
package/src/standards/token.js
DELETED
|
@@ -1,137 +0,0 @@
|
|
|
1
|
-
import Roles from './roles.js'
|
|
2
|
-
|
|
3
|
-
export default class Token extends Roles {
|
|
4
|
-
/**
|
|
5
|
-
* string
|
|
6
|
-
*/
|
|
7
|
-
#name
|
|
8
|
-
/**
|
|
9
|
-
* String
|
|
10
|
-
*/
|
|
11
|
-
#symbol
|
|
12
|
-
/**
|
|
13
|
-
* uint
|
|
14
|
-
*/
|
|
15
|
-
#holders = 0
|
|
16
|
-
/**
|
|
17
|
-
* Object => Object => uint
|
|
18
|
-
*/
|
|
19
|
-
#balances = {}
|
|
20
|
-
/**
|
|
21
|
-
* Object => Object => uint
|
|
22
|
-
*/
|
|
23
|
-
#approvals = {}
|
|
24
|
-
|
|
25
|
-
#decimals = 18
|
|
26
|
-
|
|
27
|
-
#totalSupply = BigNumber.from(0)
|
|
28
|
-
|
|
29
|
-
// this.#privateField2 = 1
|
|
30
|
-
constructor(name, symbol, decimals = 18, state) {
|
|
31
|
-
if (!name) throw new Error(`name undefined`)
|
|
32
|
-
if (!symbol) throw new Error(`symbol undefined`)
|
|
33
|
-
|
|
34
|
-
super(state?.roles)
|
|
35
|
-
|
|
36
|
-
this.#name = name
|
|
37
|
-
this.#symbol = symbol
|
|
38
|
-
this.#decimals = decimals
|
|
39
|
-
}
|
|
40
|
-
|
|
41
|
-
// enables snapshotting
|
|
42
|
-
// needs dev attention so nothing breaks after snapshot happens
|
|
43
|
-
// iow everything that is not static needs to be included in the stateObject
|
|
44
|
-
/**
|
|
45
|
-
* @return {Object} {holders, balances, ...}
|
|
46
|
-
*/
|
|
47
|
-
get state() {
|
|
48
|
-
return {
|
|
49
|
-
...super.state,
|
|
50
|
-
holders: this.holders,
|
|
51
|
-
balances: this.balances,
|
|
52
|
-
approvals: { ...this.#approvals },
|
|
53
|
-
totalSupply: this.totalSupply
|
|
54
|
-
}
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
get totalSupply() {
|
|
58
|
-
return this.#totalSupply
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
get name() {
|
|
62
|
-
return this.#name
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
get symbol() {
|
|
66
|
-
return this.#symbol
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
get holders() {
|
|
70
|
-
return this.#holders
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
get balances() {
|
|
74
|
-
return {...this.#balances}
|
|
75
|
-
}
|
|
76
|
-
|
|
77
|
-
mint(to, amount) {
|
|
78
|
-
if (!this.hasRole(msg.sender, 'MINT')) throw new Error('not allowed')
|
|
79
|
-
|
|
80
|
-
this.#totalSupply = this.#totalSupply.add(amount)
|
|
81
|
-
this.#increaseBalance(to, amount)
|
|
82
|
-
}
|
|
83
|
-
|
|
84
|
-
burn(from, amount) {
|
|
85
|
-
if (!this.hasRole(msg.sender, 'BURN')) throw new Error('not allowed')
|
|
86
|
-
|
|
87
|
-
this.#totalSupply = this.#totalSupply.sub(amount)
|
|
88
|
-
this.#decreaseBalance(from, amount)
|
|
89
|
-
}
|
|
90
|
-
|
|
91
|
-
#beforeTransfer(from, to, amount) {
|
|
92
|
-
if (!this.#balances[from] || this.#balances[from] < amount) throw new Error('amount exceeds balance')
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
#updateHolders(address, previousBalance) {
|
|
96
|
-
if (this.#balances[address].toHexString() === '0x00') this.#holders -= 1
|
|
97
|
-
else if (this.#balances[address].toHexString() !== '0x00' && previousBalance.toHexString() === '0x00') this.#holders += 1
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
#increaseBalance(address, amount) {
|
|
101
|
-
if (!this.#balances[address]) this.#balances[address] = BigNumber.from(0)
|
|
102
|
-
const previousBalance = this.#balances[address]
|
|
103
|
-
|
|
104
|
-
this.#balances[address] = this.#balances[address].add(amount)
|
|
105
|
-
this.#updateHolders(address, previousBalance)
|
|
106
|
-
}
|
|
107
|
-
|
|
108
|
-
#decreaseBalance(address, amount) {
|
|
109
|
-
const previousBalance = this.#balances[address]
|
|
110
|
-
this.#balances[address] = this.#balances[address].sub(amount)
|
|
111
|
-
this.#updateHolders(address, previousBalance)
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
balanceOf(address) {
|
|
115
|
-
return this.#balances[address]
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
setApproval(operator, amount) {
|
|
119
|
-
const owner = globalThis.msg.sender
|
|
120
|
-
if (!this.#approvals[owner]) this.#approvals[owner] = {}
|
|
121
|
-
this.#approvals[owner][operator] = amount
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
approved(owner, operator, amount) {
|
|
125
|
-
return this.#approvals[owner][operator] === amount
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
transfer(from, to, amount) {
|
|
129
|
-
// TODO: is BigNumber?
|
|
130
|
-
amount = BigNumber.from(amount)
|
|
131
|
-
this.#beforeTransfer(from, to, amount)
|
|
132
|
-
this.#decreaseBalance(from, amount)
|
|
133
|
-
this.#increaseBalance(to, amount)
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
}
|
package/src/standards/voting.js
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export default class ArtOnlineVoting {}
|