UNPKG

15.6 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3const client_common_1 = require("@neo-one/client-common");
4const utils_1 = require("@neo-one/utils");
5exports.InvalidMerkleTreeException = utils_1.makeErrorWithCode('INVALID_MERKLE_TREE', () => 'Invalid Merkle tree. (no nodes found)');
6class 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}
27const 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};
53const 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};
73const 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};
83const depthFirstSearch = (node) => {
84 const mutableHashes = [];
85 depthFirstSearchWorker(node, mutableHashes);
86 return mutableHashes;
87};
88class 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}
114exports.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"]}