npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@tradle/merkle-tree-stream

v4.1.1

Published

A stream that generates a merkle tree based on the incoming data.

Downloads

19

Readme

@tradle/merkle-tree-stream

fork of merkle-tree-stream that comes with types and CLOSE_UP.

A stream that generates a merkle tree based on the incoming data.

npm install @tradle/merkle-tree-stream

Test on Node.js

Usage

var MerkleTreeStream = require('@tradle/merkle-tree-stream')
var crypto = require('crypto')

var stream = new MerkleTreeStream({
  leaf: function (leaf, roots) {
    // this function should hash incoming data
    // roots in the current partial roots of the merkle tree
    return crypto.createHash('sha256').update(leaf.data).digest()
  },
  parent: function (a, b) {
    // hash two merkle tree node hashes into a new parent hash
    return crypto.createHash('sha256').update(a.hash).update(b.hash).digest()
  },
  // Set to true if you want the stream to write a CLOSE_UP entry before
  // the end of the stream. Read more about CLOSE_UP below.
  closeUp: false
})

stream.write('hello')
stream.write('hashed')
stream.write('world')

stream.on('data', function (data) {
  console.log(data)
})

Running the above will print

{ index: 0,
  parent: 1,
  hash: <Buffer 2c f2 4d ba 5f b0 a3 0e 26 e8 3b 2a c5 b9 e2 9e 1b 16 1e 5c 1f a7 42 5e 73 04 33 62 93 8b 98 24>,
  data: <Buffer 68 65 6c 6c 6f> } // 'hello' as buffer
{ index: 2,
  parent: 1,
  hash: <Buffer 1a 06 df 82 4e d7 41 b5 3c 78 50 79 a6 34 7f 00 ee c5 af 82 f9 85 07 75 40 9c a6 9d ff 40 68 a6>,
  data: <Buffer 68 61 68 73 65 64> } // 'hashed' as buffer
{ index: 1,
  parent: 3,
  hash: <Buffer 75 93 7d 52 c2 63 23 6d 6a 05 8e f9 c2 a6 2d d5 20 d8 c6 d0 c0 f4 a0 83 57 29 e5 97 99 25 c4 d4>,
  data: null }
{ index: 4,
  parent: 5,
  hash: <Buffer 48 6e a4 62 24 d1 bb 4f b6 80 f3 4f 7c 9a d9 6a 8f 24 ec 88 be 73 ea 8e 5a 6c 65 26 0e 9c b8 a7>,
  data: <Buffer 77 6f 72 6c 64> } // 'world' as buffer

index is the tree node index. all even numbers are data nodes (will have a non-null data property).

parent is the index of a tree node's parent node.

hash is the hash of a tree node.

You can always access the current partial roots of the merkle tree by accessing stream.roots. If the number of nodes written to the stream is not a power of 2 then stream.roots will contain more than 1 node (at most log2(number-of-nodes-written)). Otherwise it will contain just a single root.

Optionally you can also pass in an array of roots in the options map as {roots: roots} to continue adding data to a previously generated merkle tree.

Low-level interface

A non stream low-level interface can required by doing require('merkle-tree-stream/generator').

var MerkleGenerator = require('merkle-tree-stream/generator')
var gen = new MerkleGenerator({leaf: ..., parent: ...}) // same options as above

var nodes = gen.next('some data')
console.log(nodes) // returns the tree nodes generated, similar to the stream output
console.log(gen.roots) // contains the current roots nodes

Related

See mafintosh/flat-tree for more information about how node/parent indexes are calculated.

Closing up the tree and enforcing a single root

If you write an uneven number of nodes to the stream, the tree will have dangling nodes upon further examination.

const stream = new MerkleTreeStream(/* ... */)
stream.write('a')
stream.write('b')
stream.write('c')
stream.write('d')
stream.write('e')

This would cause a tree structure looking like this with the e node dangling.

      abcd
     /    \
  ab       cd
 /  \     /  \
a    b   c    d   e

Here ab is the parent which's hash is generated using options.parent(a, b).

By writing the special object MerkleTreeStream.CLOSE_UP, you can add the missing parent nodes.

stream.write(MerkleTreeStream.CLOSE_UP)
           abcdeeee
         /          \
     abcd            eeee
    /    \          /    \
  ab      cd      **      ee
 /  \    /  \    /  \    /  \
a    b  c    d  *    *  *    e

The newly created parent node ee is created using options.parent(e, e) while eeee is created using options.parent(ee, ee).

The * nodes are not written to the merkle tree.

License

MIT