forked from ethereumjs/merkle-patricia-tree
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathproof.js
95 lines (90 loc) · 3.05 KB
/
proof.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
const TrieNode = require('./trieNode')
const ethUtil = require('ethereumjs-util')
const matchingNibbleLength = require('./util').matchingNibbleLength
/**
* Returns a merkle proof for a given key
* @method Trie.prove
* @param {Trie} trie
* @param {String} key
* @param {Function} cb A callback `Function` (arguments {Error} `err`, {Array.<TrieNode>} `proof`)
*/
exports.prove = function (trie, key, cb) {
var nodes
trie.findPath(key, function (err, node, remaining, stack) {
if (err) return cb(err)
if (remaining.length > 0) return cb(new Error('Node does not contain the key'))
nodes = stack
var p = []
for (var i = 0; i < nodes.length; i++) {
var rlpNode = nodes[i].serialize()
if ((rlpNode.length >= 32) || (i === 0)) {
p.push(rlpNode)
}
}
cb(null, p)
})
}
/**
* Verifies a merkle proof for a given key
* @method Trie.verifyProof
* @param {Buffer} rootHash
* @param {String} key
* @param {Array.<TrieNode>} proof
* @param {Function} cb A callback `Function` (arguments {Error} `err`, {String} `val`)
*/
exports.verifyProof = function (rootHash, key, proof, cb) {
key = TrieNode.stringToNibbles(key)
var wantHash = ethUtil.toBuffer(rootHash)
for (var i = 0; i < proof.length; i++) {
var p = ethUtil.toBuffer(proof[i])
var hash = ethUtil.sha3(proof[i])
if (Buffer.compare(hash, wantHash)) {
return cb(new Error('Bad proof node ' + i + ': hash mismatch'))
}
var node = new TrieNode(ethUtil.rlp.decode(p))
var cld
if (node.type === 'branch') {
if (key.length === 0) {
if (i !== proof.length - 1) {
return cb(new Error('Additional nodes at end of proof (branch)'))
}
return cb(null, node.value)
}
cld = node.raw[key[0]]
key = key.slice(1)
if (cld.length === 2) {
var embeddedNode = new TrieNode(cld)
if (i !== proof.length - 1) {
return cb(new Error('Additional nodes at end of proof (embeddedNode)'))
}
if (matchingNibbleLength(embeddedNode.key, key) !== embeddedNode.key.length) {
return cb(new Error('Key length does not match with the proof one (embeddedNode)'))
}
key = key.slice(embeddedNode.key.length)
if (key.length !== 0) {
return cb(new Error('Key does not match with the proof one (embeddedNode)'))
}
return cb(null, embeddedNode.value)
} else {
wantHash = cld
}
} else if ((node.type === 'extention') || (node.type === 'leaf')) {
if (matchingNibbleLength(node.key, key) !== node.key.length) {
return cb(new Error('Key does not match with the proof one (extention|leaf)'))
}
cld = node.value
key = key.slice(node.key.length)
if (key.length === 0) {
if (i !== proof.length - 1) {
return cb(new Error('Additional nodes at end of proof (extention|leaf)'))
}
return cb(null, cld)
} else {
wantHash = cld
}
} else {
return cb(new Error('Invalid node type'))
}
}
cb(new Error('Unexpected end of proof'))
}