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 🙏

© 2026 – Pkg Stats / Ryan Hefner

fingerprint.dev

v0.1.1

Published

Browser fingerprint collection library for visitor identification

Readme

fingerprint.dev

Browser client for fingerprint.dev. Collects browser-side signals, hashes them locally, and returns a stable visitor_id from the hosted /v1/identify endpoint.

Install

npm install fingerprint.dev

Quickstart

import { identify } from "fingerprint.dev";

let result = await identify({ apiKey: "fp_live_..." });

result.visitor_id;      // string | null  — stable per visitor
result.confidence;      // number | null  — 0.0 to 1.0
result.kind;            // "new" | "match"
result.request_id;      // string         — "req_..."
result.signals;         // server-side JA4 fingerprints
result.browser_signals; // hashed browser signals sent up

identify() throws Identify.Error on failure. The code field is one of INVALID_CONFIG, API_ERROR, TIMEOUT, NETWORK_ERROR, INVALID_RESPONSE, UNKNOWN_ERROR.

import { identify, Identify } from "fingerprint.dev";

try {
  var result = await identify({ apiKey: "fp_live_..." });
} catch (err) {
  if (err instanceof Identify.Error && err.code === "TIMEOUT") {
    // retry, fall back, etc.
  }
  throw err;
}

Options

| Option | Type | Default | Notes | | ---------- | --------- | ----------------------------- | ------------------------------------------------------ | | apiKey | string | — | Required. fp_live_* or fp_test_*. | | endpoint | string | https://api.fingerprint.dev | /v1/identify is appended automatically if missing. | | timeout | number | 10000 | Milliseconds. Aborts the request; throws TIMEOUT. | | debug | boolean | false | Logs request progress and signal availability. |

Signal semantics

The client collects six browser signals: canvas, webgl, audio, screen, fonts, navigator. Each is string | nullnull means the browser API was blocked, unavailable, or failed. Null is preserved through hashing so unavailable APIs read as missing evidence rather than a stable pseudo-signal.

import { collectSignals, hashSignals } from "fingerprint.dev";

let raw = await collectSignals();    // Identify.RawSignals — debug/inspect only
let hashed = await hashSignals(raw); // Identify.HashedSignals — what gets sent

if (hashed.webgl === null) {
  // WebGL was unavailable or blocked.
}

Raw signal values never leave the browser. Only SHA-256 hashes (truncated to 12 hex chars) are sent. debug: true logs request progress and per-signal availability — it never logs raw or hashed signal values.

If crypto.subtle is unavailable (insecure context, ancient browser), identify() skips signal collection entirely and still calls the API so the server can produce a visitor id from connection-level fingerprints.

Exports

  • identify(options) — collect, hash, send, return result
  • collectSignals() — raw signals only (no network)
  • hashSignals(raw) — hash without sending
  • Identify — type namespace + the Error class
    • Identify.Error — thrown on failure; has code and optional statusCode
    • Types: Identify.Options, Identify.Result, Identify.Response, Identify.RawSignals, Identify.HashedSignals, Identify.SignalValue, Identify.Fingerprints