Skip to content

Preeternal/react-native-file-hash

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

81 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

@preeternal/react-native-file-hash

Native streaming hashes for React Native files, strings, HMAC, XXH3, and BLAKE3.

npm version npm downloads

Use it when your app needs to verify large downloads, fingerprint media, deduplicate local files or cached uploads, generate fast cache keys, compare local content, or authenticate data with HMAC or keyed BLAKE3.

Hash multi-GB files without loading them into JavaScript. Native code streams files in chunks, keeps the work off the JS thread, and returns a lowercase hex digest. Large inputs do not cross the JS bridge or sit in JS memory, which helps avoid memory spikes and OOM crashes.

Highlights

  • Streams file data from disk instead of loading the whole file into JS memory.
  • Uses native background work on iOS and Android, keeping the UI responsive.
  • Safe for concurrent calls: each operation owns its native hash state.
  • Defaults to SHA-256 when you do not pass an algorithm.
  • Supports AbortController cancellation for long-running file hashes.
  • Supports local files, Android content:// URIs, and iOS provider-backed file URLs when the app has access.
  • Returns lowercase hex strings for every algorithm.
  • Includes native-side SHA variants, HMAC algorithms, XXH3, BLAKE3, and keyed BLAKE3.

Installation

npm install @preeternal/react-native-file-hash

or:

yarn add @preeternal/react-native-file-hash

or:

bun add @preeternal/react-native-file-hash

React Native 0.60+ autolinks the native module.

For iOS, install pods:

cd ios
bundle exec pod install

Expo

Expo Go is not supported because this package contains native code. Use a Development Build, EAS Build, or expo prebuild.

For the default native engine, no plugin options are required:

{
  "expo": {
    "plugins": ["@preeternal/react-native-file-hash"]
  }
}

Quick Start

import { fileHash, stringHash } from '@preeternal/react-native-file-hash';

const digest = await fileHash('file:///path/to/video.mp4');
// SHA-256 lowercase hex by default

const xxh3 = await fileHash('file:///path/to/video.mp4', {
  algorithm: 'XXH3-64',
});

const textDigest = await stringHash('hello world');

For large payloads, prefer fileHash. stringHash is intended for small strings or small base64 payloads already in JS memory.

Cancel Long Hashes

Cancellation follows the same shape as fetch and many HTTP clients:

import { fileHash } from '@preeternal/react-native-file-hash';

const controller = new AbortController();

const promise = fileHash(fileUri, {
  algorithm: 'SHA-256',
  signal: controller.signal,
});

controller.abort();

try {
  await promise;
} catch (error) {
  if (
    error instanceof Error &&
    error.name === 'AbortError' &&
    (error as { code?: string }).code === 'E_CANCELLED'
  ) {
    // Hash was cancelled.
  }
}

Cancellation is cooperative. Large file hashes stop at chunk boundaries. Small stringHash calls may finish before the abort is observed.

HMAC And Keyed BLAKE3

HMAC is selected by the algorithm name:

const hmac = await fileHash(fileUri, {
  algorithm: 'HMAC-SHA-256',
  hashOptions: {
    key: 'super-secret',
    keyEncoding: 'utf8',
  },
});

Keyed BLAKE3 uses the regular BLAKE3 algorithm with a key. The decoded key must be 32 bytes:

const keyed = await fileHash(fileUri, {
  algorithm: 'BLAKE3',
  hashOptions: {
    key: '3031323334353637383961626364656630313233343536373839616263646566',
    keyEncoding: 'hex',
  },
});

keyEncoding can be 'utf8', 'hex', or 'base64'. The default is 'utf8'. For BLAKE3, hashOptions.key is optional: omit it for regular BLAKE3, pass it only when you want keyed BLAKE3.

HMAC algorithms require hashOptions.key. If you intentionally need HMAC with an empty key, pass key: '' explicitly; omitting key is rejected.

Seeded XXH3

XXH3-64 and XXH3-128 support an optional unsigned 64-bit seed. Omit it for regular unseeded XXH3. A seed is not a secret and does not make XXH3 cryptographic. It selects a reproducible XXH3 namespace: the same bytes, algorithm, and seed produce the same digest on your backend, CLI tools, and mobile app.

With the optional zig engine, seeded XXH3 currently means XXH3-64; XXH3-128 is available only in the default native engine.

The most direct use case is server-side verification. For example, your backend can publish a download manifest with both the expected XXH3 digest and the seed that was used to compute it:

import { fileHash } from '@preeternal/react-native-file-hash';

const manifest = {
  url: 'https://example.com/assets/video.mp4',
  xxh3Seed: '12345678901234567890',
  xxh3: '4b5e0a417dfa7ed2fb965bc17c16bd34',
};

const actual = await fileHash(localFileUri, {
  algorithm: 'XXH3-128',
  hashOptions: {
    seed: manifest.xxh3Seed,
  },
});

if (actual !== manifest.xxh3) {
  throw new Error('Downloaded file failed checksum verification');
}

Pass large u64 seeds from a backend as strings, either decimal or 0x hex. JavaScript number is safe only up to Number.MAX_SAFE_INTEGER; values like 12345678901234567890 must be passed as a string or bigint so they are not rounded before hashing.

For app-owned namespaces, you can derive a stable seed from a readable label:

import {
  fileHash,
  xxh3SeedFromLabel,
} from '@preeternal/react-native-file-hash';

const mediaCacheSeed = xxh3SeedFromLabel('media-cache-v1');

const cacheKey = await fileHash(fileUri, {
  algorithm: 'XXH3-128',
  hashOptions: {
    seed: mediaCacheSeed,
  },
});

hashOptions.seed accepts a bigint, a non-negative safe integer, a decimal string, or a 0x hex string. Values are normalized before native hashing, so 12345, 12345n, and '0x3039' all use the same seed.

xxh3SeedFromLabel(label) derives a deterministic seed from a UTF-8 label using FNV-1a 64-bit. The helper returns a canonical 0x hex seed, for example 0x091677a156a7756e. Use it when the app controls the namespace, such as media-cache-v1 or upload-dedupe-v2. If a backend or CLI must reproduce the same hashes, share either the derived seed value or the exact label derivation rule. Use HMAC or keyed BLAKE3 when authenticity matters.

API

fileHash(filePath, request?)

Hashes a file by streaming it from native code.

const mac = await fileHash(fileUri, {
  algorithm: 'HMAC-SHA-256',
  hashOptions: {
    key: 'upload-signing-secret',
    keyEncoding: 'utf8',
  },
  signal: abortController.signal,
});

filePath can be:

  • a local path or file:// URI;
  • an Android content:// URI, for example from the system document picker;
  • an iOS provider-backed file URL, for example from Files or iCloud, when the app has access.

If request.algorithm is omitted, SHA-256 is used. Use hashOptions.key only with HMAC algorithms or keyed BLAKE3; regular hashes reject keys.

stringHash(text, request?)

Hashes a small JS string or base64 payload.

const digest = await stringHash('hello world', {
  algorithm: 'SHA-256',
  encoding: 'utf8',
});

const fromBase64 = await stringHash(base64Payload, {
  algorithm: 'BLAKE3',
  encoding: 'base64',
});

If request.algorithm is omitted, SHA-256 is used. If request.encoding is omitted, utf8 is used.

Request Types

type HashRequest = {
  algorithm?: THashAlgorithm;
  hashOptions?: HashOptions;
  signal?: HashAbortSignal;
};

type StringHashRequest = HashRequest & {
  encoding?: 'utf8' | 'base64';
};

type HashOptions = {
  key?: string;
  keyEncoding?: 'utf8' | 'hex' | 'base64';
  seed?: bigint | number | string;
};

The exported HashAbortSignal type is intentionally compatible with AbortController.signal.

Deprecated Call Forms

The positional overloads still work, but the object-style request API is the recommended form because it avoids placeholder arguments and supports cancellation cleanly.

// Deprecated, still supported for migration.
await fileHash(fileUri, 'SHA-256');
await stringHash('hello', 'SHA-256', 'utf8');

// Recommended.
await fileHash(fileUri, { algorithm: 'SHA-256' });
await stringHash('hello', { algorithm: 'SHA-256', encoding: 'utf8' });

hashString(...) is a deprecated alias for stringHash(...) and will be removed in a future major release.

Algorithms

Algorithm Use case Notes
SHA-256 Default general-purpose cryptographic hash Good default for integrity checks
SHA-384, SHA-512 Stronger SHA-2 variants Larger output, usually slower
SHA-224, SHA-512/224, SHA-512/256 SHA-2 compatibility variants Useful for protocols requiring these exact digests
MD5, SHA-1 Legacy compatibility Do not use for new security-sensitive designs
HMAC-SHA-256 Shared-secret authentication Good default HMAC choice
HMAC-SHA-224/384/512, HMAC-MD5, HMAC-SHA-1 Protocol compatibility Prefer SHA-256+ for new designs
XXH3-64, XXH3-128 Fast non-cryptographic checksums Supports optional seed; not authentication
BLAKE3 Modern high-performance hash Also supports keyed mode with a 32-byte key

Output Lengths

Algorithm Output length
MD5, HMAC-MD5 16 bytes, 32 hex chars
SHA-1, HMAC-SHA-1 20 bytes, 40 hex chars
SHA-224, HMAC-SHA-224 28 bytes, 56 hex chars
SHA-256, HMAC-SHA-256, SHA-512/256, BLAKE3 32 bytes, 64 hex chars
SHA-384, HMAC-SHA-384 48 bytes, 96 hex chars
SHA-512, HMAC-SHA-512 64 bytes, 128 hex chars
SHA-512/224 28 bytes, 56 hex chars
XXH3-64 8 bytes, 16 hex chars
XXH3-128 16 bytes, 32 hex chars

Error Handling

Common error codes:

Code Meaning
E_CANCELLED Operation was cancelled through AbortController
E_INVALID_ARGUMENT Invalid algorithm/options combination
E_INVALID_KEY Key cannot be decoded or has the wrong length
E_INVALID_INPUT Invalid string input, usually malformed base64
E_FILE_NOT_FOUND File cannot be opened
E_UNSUPPORTED_ALGORITHM Algorithm is not available in the selected engine
E_HASH_FAILED / E_FILE_HASH_FAILED Native hashing failed unexpectedly

Key rules:

  • HMAC algorithms require hashOptions.key; pass key: '' explicitly for an empty HMAC key.
  • BLAKE3 uses keyed mode only when hashOptions.key is provided.
  • BLAKE3 keyed mode requires a 32-byte key after decoding.
  • Other algorithms reject hashOptions.key.
  • hashOptions.seed is only valid for XXH3-64 and XXH3-128.
  • HashOptions.mode was removed and is rejected with E_INVALID_ARGUMENT.

Runtime Info

import {
  getRuntimeDiagnostics,
  getRuntimeInfo,
} from '@preeternal/react-native-file-hash';

const info = await getRuntimeInfo();
// { engine: 'native' } or { engine: 'zig' }

const diagnostics = await getRuntimeDiagnostics();

getRuntimeDiagnostics() includes Zig ABI/core metadata when the Zig engine is selected. For the default native engine, consumers usually do not need it.

Optional: Engine Selection

This library ships with two build-time engines:

  • native: the default engine, recommended for most apps.
  • zig: an optional engine built on the bundled zig-files-hash core.

The native engine uses platform implementations plus native C for BLAKE3 and XXH3. The zig engine can be useful when you want one portable hashing core shared across bindings, when you want to validate behavior against the Zig implementation, or when current benchmarks favor Zig for the algorithms you use. See BENCHMARKS.md for the latest engine comparison.

Package users do not need a local Zig toolchain; release artifacts include Zig prebuilts.

The selected engine is resolved at build time. The unused engine is not linked into the final native binary.

Android

Set this in your app's android/gradle.properties:

react_native_file_hash_engine=zig

If the property is omitted, native is used.

Android Zig currently routes SHA-224, SHA-256, HMAC-SHA-224, and HMAC-SHA-256 through the native pipeline in the shipped generic prebuilt setup to avoid ARM SHA-2 latency cliffs.

iOS

Set this in your app's ios/Podfile before pod install:

ENV['ZFH_ENGINE'] ||= 'zig'

If ZFH_ENGINE is omitted, native is used.

Expo Engine Selection

{
  "expo": {
    "plugins": [["@preeternal/react-native-file-hash", { "engine": "zig" }]]
  }
}

If engine is omitted, native is used.

Engine Compatibility Notes

  • XXH3-128 is currently available only in the native engine.

Performance

Use physical devices and Release builds for performance claims. Debug, simulator, and emulator runs are useful for smoke checks, but they do not represent production throughput.

Full current measurements live in BENCHMARKS.md.

Practical guidance:

  • Start with the default native engine for most apps.
  • Consider zig when you care about specific algorithms or want a portable hashing core; check BENCHMARKS.md for current device data.
  • Use SHA-256 for general integrity checks.
  • Use XXH3-64 or XXH3-128 for fast non-security checksums.
  • Use HMAC-SHA-256 for shared-secret authentication.
  • Avoid MD5 and SHA-1 for new security-sensitive designs.

Android 16 KB Page Size

The Android build enables 16 KB page alignment when the linker supports -Wl,-z,max-page-size=16384, matching Google Play requirements for newer 16 KB page size devices. Tested with NDK 27.1.x.

If your NDK/toolchain does not recognize the flag, the build continues without 16 KB alignment. Upgrade the NDK to produce 16 KB-aligned binaries.

Native Implementations

Licenses: xxHash is BSD 2-Clause; the C implementation of BLAKE3 is CC0 (public domain). See third_party/xxhash/LICENSE and third_party/blake3/LICENSE_CC0, plus accompanying Apache-2.0 notices in third_party/blake3.

Contributing

Contributions are welcome.

License

MIT. See LICENSE for details.

Made with create-react-native-library.