1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | const client_common_1 = require("@neo-one/client-common");
|
4 | const utils_1 = require("@neo-one/utils");
|
5 | exports.InvalidMerkleTreeException = utils_1.makeErrorWithCode('INVALID_MERKLE_TREE', () => 'Invalid Merkle tree. (no nodes found)');
|
6 | class MerkleTreeNode {
|
7 | constructor({ hash, leftChild, rightChild, parent, }) {
|
8 | this.hash = hash;
|
9 | this.leftChild = leftChild;
|
10 | this.rightChild = rightChild;
|
11 | this.parent = parent;
|
12 | }
|
13 | clone(parent) {
|
14 | const self = new MerkleTreeNode({
|
15 | hash: this.hash,
|
16 | parent,
|
17 | });
|
18 | if (this.leftChild !== undefined) {
|
19 | self.leftChild = this.leftChild.clone(self);
|
20 | }
|
21 | if (this.rightChild !== undefined) {
|
22 | self.rightChild = this.rightChild.clone(self);
|
23 | }
|
24 | return self;
|
25 | }
|
26 | }
|
27 | const build = (leavesIn) => {
|
28 | const leaves = leavesIn;
|
29 | if (leaves.length === 0) {
|
30 | throw new exports.InvalidMerkleTreeException();
|
31 | }
|
32 | if (leaves.length === 1) {
|
33 | return leaves[0];
|
34 | }
|
35 | const parents = [];
|
36 | const length = Math.floor((leaves.length + 1) / 2);
|
37 | for (let i = 0; i < length; i += 1) {
|
38 | const leftChild = leaves[i * 2];
|
39 | const rightChild = i * 2 + 1 === leaves.length ? leftChild : leaves[i * 2 + 1];
|
40 | const node = new MerkleTreeNode({
|
41 | hash: client_common_1.crypto.hash256(Buffer.concat([client_common_1.common.uInt256ToBuffer(leftChild.hash), client_common_1.common.uInt256ToBuffer(rightChild.hash)])),
|
42 | leftChild,
|
43 | rightChild,
|
44 | });
|
45 | parents[i] = node;
|
46 | leaves[i * 2].parent = node;
|
47 | if (i * 2 + 1 !== leaves.length) {
|
48 | leaves[i * 2 + 1].parent = node;
|
49 | }
|
50 | }
|
51 | return build(parents);
|
52 | };
|
53 | const trim = (node, index, depth, flags) => {
|
54 | const { leftChild, rightChild } = node;
|
55 | if (depth === 1 || leftChild === undefined) {
|
56 | return;
|
57 | }
|
58 | if (depth === 2) {
|
59 | if (!flags[index * 2] && !flags[index * 2 + 1]) {
|
60 | node.leftChild = undefined;
|
61 | node.rightChild = undefined;
|
62 | }
|
63 | }
|
64 | else if (rightChild !== undefined) {
|
65 | trim(leftChild, index * 2, depth - 1, flags);
|
66 | trim(rightChild, index * 2 + 1, depth - 1, flags);
|
67 | if (leftChild.leftChild === undefined && rightChild.rightChild === undefined) {
|
68 | node.leftChild = undefined;
|
69 | node.rightChild = undefined;
|
70 | }
|
71 | }
|
72 | };
|
73 | const depthFirstSearchWorker = (node, mutableHashes) => {
|
74 | const { leftChild, rightChild } = node;
|
75 | if (leftChild === undefined || rightChild === undefined) {
|
76 | mutableHashes.push(node.hash);
|
77 | }
|
78 | else {
|
79 | depthFirstSearchWorker(leftChild, mutableHashes);
|
80 | depthFirstSearchWorker(rightChild, mutableHashes);
|
81 | }
|
82 | };
|
83 | const depthFirstSearch = (node) => {
|
84 | const mutableHashes = [];
|
85 | depthFirstSearchWorker(node, mutableHashes);
|
86 | return mutableHashes;
|
87 | };
|
88 | class MerkleTree {
|
89 | static computeRoot(hashes) {
|
90 | const tree = new this(hashes);
|
91 | return tree.root.hash;
|
92 | }
|
93 | constructor(hashesOrNode) {
|
94 | this.root = Array.isArray(hashesOrNode)
|
95 | ? build(hashesOrNode.map((hash) => new MerkleTreeNode({ hash })))
|
96 | : hashesOrNode;
|
97 | this.depth = 1;
|
98 | for (let node = this.root; node.leftChild !== undefined; node = node.leftChild) {
|
99 | this.depth += 1;
|
100 | }
|
101 | }
|
102 | trim(flags) {
|
103 | const result = this.root.clone();
|
104 | trim(result, 0, this.depth, flags);
|
105 | return new MerkleTree(result);
|
106 | }
|
107 | depthFirstSearch() {
|
108 | return depthFirstSearch(this.root);
|
109 | }
|
110 | toHashArray() {
|
111 | return this.depthFirstSearch();
|
112 | }
|
113 | }
|
114 | exports.MerkleTree = MerkleTree;
|
115 |
|
116 | //# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["MerkleTree.ts"],"names":[],"mappings":"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","file":"neo-one-node-core/src/crypto/MerkleTree.js","sourcesContent":["// tslint:disable readonly-keyword no-object-mutation no-array-mutation\nimport { common, crypto, UInt256 } from '@neo-one/client-common';\nimport { makeErrorWithCode } from '@neo-one/utils';\n\nexport const InvalidMerkleTreeException = makeErrorWithCode(\n  'INVALID_MERKLE_TREE',\n  () => 'Invalid Merkle tree. (no nodes found)',\n);\n\nclass MerkleTreeNode {\n  public readonly hash: UInt256;\n  public leftChild: MerkleTreeNode | undefined;\n  public rightChild: MerkleTreeNode | undefined;\n  public parent: MerkleTreeNode | undefined;\n\n  public constructor({\n    hash,\n    leftChild,\n    rightChild,\n    parent,\n  }: {\n    readonly hash: UInt256;\n    readonly leftChild?: MerkleTreeNode;\n    readonly rightChild?: MerkleTreeNode;\n    readonly parent?: MerkleTreeNode;\n  }) {\n    this.hash = hash;\n    this.leftChild = leftChild;\n    this.rightChild = rightChild;\n    this.parent = parent;\n  }\n\n  public clone(parent?: MerkleTreeNode): MerkleTreeNode {\n    const self = new MerkleTreeNode({\n      hash: this.hash,\n      parent,\n    });\n\n    if (this.leftChild !== undefined) {\n      self.leftChild = this.leftChild.clone(self);\n    }\n    if (this.rightChild !== undefined) {\n      self.rightChild = this.rightChild.clone(self);\n    }\n\n    return self;\n  }\n}\n\nconst build = (leavesIn: readonly MerkleTreeNode[]): MerkleTreeNode => {\n  const leaves = leavesIn;\n  if (leaves.length === 0) {\n    throw new InvalidMerkleTreeException();\n  }\n  if (leaves.length === 1) {\n    return leaves[0];\n  }\n\n  const parents = [];\n  const length = Math.floor((leaves.length + 1) / 2);\n  // tslint:disable-next-line no-loop-statement\n  for (let i = 0; i < length; i += 1) {\n    const leftChild = leaves[i * 2];\n\n    const rightChild = i * 2 + 1 === leaves.length ? leftChild : leaves[i * 2 + 1];\n    const node = new MerkleTreeNode({\n      hash: crypto.hash256(\n        Buffer.concat([common.uInt256ToBuffer(leftChild.hash), common.uInt256ToBuffer(rightChild.hash)]),\n      ),\n      leftChild,\n      rightChild,\n    });\n\n    parents[i] = node;\n    leaves[i * 2].parent = node;\n\n    if (i * 2 + 1 !== leaves.length) {\n      leaves[i * 2 + 1].parent = node;\n    }\n  }\n\n  return build(parents);\n};\n\nconst trim = (node: MerkleTreeNode, index: number, depth: number, flags: readonly boolean[]) => {\n  const { leftChild, rightChild } = node;\n  if (depth === 1 || leftChild === undefined) {\n    return;\n  }\n\n  if (depth === 2) {\n    if (!flags[index * 2] && !flags[index * 2 + 1]) {\n      node.leftChild = undefined;\n      node.rightChild = undefined;\n    }\n  } else if (rightChild !== undefined) {\n    trim(leftChild, index * 2, depth - 1, flags);\n    trim(rightChild, index * 2 + 1, depth - 1, flags);\n    if (leftChild.leftChild === undefined && rightChild.rightChild === undefined) {\n      node.leftChild = undefined;\n      node.rightChild = undefined;\n    }\n  }\n};\n\nconst depthFirstSearchWorker = (node: MerkleTreeNode, mutableHashes: UInt256[]): void => {\n  const { leftChild, rightChild } = node;\n  if (leftChild === undefined || rightChild === undefined) {\n    mutableHashes.push(node.hash);\n  } else {\n    depthFirstSearchWorker(leftChild, mutableHashes);\n    depthFirstSearchWorker(rightChild, mutableHashes);\n  }\n};\n\nconst depthFirstSearch = (node: MerkleTreeNode): readonly UInt256[] => {\n  const mutableHashes: UInt256[] = [];\n  depthFirstSearchWorker(node, mutableHashes);\n\n  return mutableHashes;\n};\n\nexport class MerkleTree {\n  public static computeRoot(hashes: readonly UInt256[]): UInt256 {\n    const tree = new this(hashes);\n\n    return tree.root.hash;\n  }\n\n  public readonly root: MerkleTreeNode;\n  public readonly depth: number;\n\n  public constructor(hashesOrNode: readonly UInt256[] | MerkleTreeNode) {\n    this.root = Array.isArray(hashesOrNode)\n      ? build(hashesOrNode.map((hash) => new MerkleTreeNode({ hash })))\n      : (hashesOrNode as MerkleTreeNode);\n    this.depth = 1;\n    // tslint:disable-next-line no-loop-statement no-let\n    for (let node = this.root; node.leftChild !== undefined; node = node.leftChild) {\n      this.depth += 1;\n    }\n  }\n\n  public trim(flags: readonly boolean[]): MerkleTree {\n    const result = this.root.clone();\n    trim(result, 0, this.depth, flags);\n\n    return new MerkleTree(result);\n  }\n\n  public depthFirstSearch(): readonly UInt256[] {\n    return depthFirstSearch(this.root);\n  }\n\n  public toHashArray(): readonly UInt256[] {\n    return this.depthFirstSearch();\n  }\n}\n"]}
|