-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathindex.js
99 lines (88 loc) · 2.78 KB
/
index.js
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
import fs from 'fs';
import pkg0 from 'ethers';
const { ethers } = pkg0;
import pkg1 from 'merkletreejs';
const { MerkleTree } = pkg1;
function generateLeaf(address) {
const encodedData = ethers.utils
.keccak256(
ethers.utils.defaultAbiCoder.encode(
['address'],
[ethers.utils.getAddress(address)]
)
)
.slice(2);
return Buffer.from(encodedData, 'hex');
}
function generateMerkleTree(allowlist) {
return new MerkleTree(allowlist.map(generateLeaf), ethers.utils.keccak256, {
sortPairs: true,
});
}
function verifyMerkleProof(merkleRoot, address, merkleProof) {
return MerkleTree.verify(
merkleProof,
generateLeaf(address),
merkleRoot,
ethers.utils.keccak256,
{ sortPairs: true }
);
}
function getMerkleProof(tree, address, index) {
const leaf = generateLeaf(address);
return tree.getHexProof(leaf, index);
}
function getEncodedMerkleProof(tree, address, allowlist) {
const index = allowlist.findIndex((a) => a === address);
if (index < 0) {
throw new Error(`${address} is not on the allowlist`);
}
return ethers.utils.defaultAbiCoder.encode(
['bytes32[]'],
[getMerkleProof(tree, address, index)]
);
}
function fetchAllowlistAddresses(filepath) {
const fileContent = fs.readFileSync(filepath, { encoding: 'utf-8' });
const addresses = fileContent
.split('\n')
.map((e) => e.trim())
.filter((e) => e !== '')
.map((address) => ethers.utils.getAddress(address));
return addresses;
}
const allowlist = fetchAllowlistAddresses('./allowlist.csv');
// Build tree and root
const merkleTree = generateMerkleTree(allowlist);
console.log(`Merkle tree successfully generated.`);
const merkleRoot = '0x' + merkleTree.getRoot().toString('hex');
console.log(`Merkle root: ${merkleRoot}`);
console.log(`-------------------`);
// Generate and verify proof for test address
const testAddress = allowlist[0];
const merkleProof = getMerkleProof(merkleTree, testAddress, allowlist);
const isVerified = verifyMerkleProof(merkleRoot, testAddress, merkleProof);
console.log(`Test address: ${testAddress}`);
console.log(`Test proof: ${merkleProof}`);
console.log(`Test proof is verified: ${isVerified}`);
console.log(`-------------------`);
// Generate encoded proof to supply to contract
const encodedMerkleProof = getEncodedMerkleProof(
merkleTree,
testAddress,
allowlist
);
console.log(`Encoded test proof (for contract): ${encodedMerkleProof}`);
console.log(`-------------------`);
// Generate JSON file
console.log('Generating tree.json ...');
const treeJSON = { root: merkleRoot, proofs: {} };
for (let address of allowlist) {
treeJSON.proofs[address] = getEncodedMerkleProof(
merkleTree,
address,
allowlist
);
}
fs.writeFileSync('tree.json', JSON.stringify(treeJSON, null, 4));
console.log('tree.json generated');