highwayhasher
v0.4.4
Published
HighwayHash implementation for both node and browsers
Downloads
616
Maintainers
Readme
HighwayHasher
HighwayHasher is JS bindings to the Rust implementation of Google's HighwayHash, a fast, keyed and strong hash function.
Features
- ✔ Isomorphic: Run on node and in the browser with the same API
- ✔ Fast: Generate hashes at over 2 GB/s when running on SIMD enabled web assembly
- ✔ Faster: Generate hashes at over 8 GB/s when running on native hardware
- ✔ Self-contained: Zero runtime dependencies
- ✔ Accessible: Prebuilt native modules means no build dependencies
- ✔ Completeness: Generate 64, 128, and 256bit hashes
- ✔ Incremental: Hash data incrementally
- ✔ Small: Less than 10 KB when minified and gzipped (or 5 KB when using the slim entrypoint)
Install
npm i highwayhasher
Quick Start
const { HighwayHash } = require("highwayhasher");
// or import { HighwayHash } from "highwayhasher";
// The hash is based on a 32 byte key (optional)
const keyData = Uint8Array.from(new Array(32).fill(1));
// Load the hash function (compiles and caches the wasm if on browser).
const hash = await HighwayHash.load(keyData);
// Append any amount of data
hash.append(Uint8Array.from([0]));
// When all the data is appended, call the hash you want.
// Note, do not call any additional functions after finalization
let out = hash.finalize64();
let expected = Uint8Array.from([120, 221, 205, 199, 170, 67, 171, 126]);
expect(out).toEqual(expected);
That's it! Now your program will use native code on nodejs and Wasm in the browser!
Quick Start (browser only)
To get started without a build tool, the easiest way to reference highwayhasher in the browser is through a CDN:
<body>
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/browser-fat/umd/index_browser_fat.min.js"></script>
<script>(function() {
const { HighwayHash } = highwayhasher;
const keyData = Uint8Array.from(new Array(32).fill(1));
HighwayHash.loadModule().then((hasher) => {
const out = hasher.hash64(keyData, Uint8Array.from([0]));
console.log(out);
});
})();
</script>
</body>
Or one can reference the slim endpoint to load highwayhasher more efficiently:
<script type="module">
import { HighwayHash } from 'https://cdn.jsdelivr.net/npm/[email protected]/dist/browser-slim/es/index_browser_slim.min.js';
const hasher = await HighwayHash.loadModule({
wasm: {
simd: 'https://cdn.jsdelivr.net/npm/[email protected]/dist/highwayhasher_wasm_simd_bg.wasm',
sisd: 'https://cdn.jsdelivr.net/npm/[email protected]/dist/highwayhasher_wasm_bg.wasm',
}
});
const keyData = Uint8Array.from(new Array(32).fill(1));
const out = hasher.hash64(keyData, Uint8Array.from([0]));
console.log(out);
</script>
Non-Streaming API
When the data to hash is known entirely, one can use one of the non-streaming APIs that operate more efficiently than their incremental counterparts on small data.
import { HighwayHash } from "highwayhasher";
const keyData = Uint8Array.from(new Array(32).fill(1));
const hasher = await HighwayHash.loadModule();
const out = hasher.hash64(keyData, Uint8Array.from([0]));
Wasm-Only
Since Wasm is cross platform, one can drop any reliance on native dependencies by opting to only use the Wasm implementation
const { WasmHighwayHash: HighwayHash } = require("highwayhasher");
const keyData = Uint8Array.from(new Array(32).fill(1));
const hash = await HighwayHash.load(keyData);
hash.append(Uint8Array.from([0]));
let out = hash.finalize64();
let expected = Uint8Array.from([120, 221, 205, 199, 170, 67, 171, 126]);
expect(out).toEqual(expected);
Do note that there is a max number to concurrent Wasm hashers. It is currently 372 instances. If this is believed to be a significant limitation, open an issue so we can discuss it!
Slim Module
By default, the highwayhasher
entrypoint includes Wasm that is base64 inlined. This is the default as most developers will probably not need to care. However some developers will care: those running the library in environments where Wasm is executable but not compilable, or those who are ambitious about reducing compute and bandwidth costs for their users.
To cater to these use cases, there is a highwayhasher/slim
package that operates the exactly the same, except now it is expected for developers to prime initialization through some other means.
If you know the environment has Wasm SIMD enabled (for instance, deploying on Cloudflare workers):
import { HighwayHash } from "highwayhasher/slim";
import wasm from "highwayhasher/simd.wasm";
const hasher = await HighwayHash.loadModule({ wasm });
If runtime detection of SIMD is preferred, both SIMD disabled and enabled modules can be referenced.
import { HighwayHash } from "highwayhasher/slim";
import simd from "highwayhasher/simd.wasm";
import sisd from "highwayhasher/sisd.wasm";
const hasher = await HighwayHash.loadModule({
wasm: {
simd,
sisd,
}
});