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

hybrid-id

v1.0.0

Published

Compact, time-sortable unique ID generator for Node.js. Drop-in UUID/ULID alternative with base62 encoding, Stripe-style prefixes, and configurable entropy profiles.

Readme

hybrid-id

Compact, time-sortable unique identifiers for Node.js

npm version CI types license

A space-efficient alternative to UUID with configurable entropy profiles, Stripe-style prefixes, and an instance-based API. Generate chronologically sortable, URL-safe identifiers 33–56% smaller than canonical UUIDs — with zero runtime dependencies, full TypeScript types, and dual ESM/CJS output.

Node port of alesitom/hybrid-id (PHP). Compatibility: spec parity — same format, layout, and parsing as the PHP library, so a non-blind ID generated in PHP parses in Node and vice versa. It is not byte-exact across languages (random bytes differ). The one exception is UUID conversion, which is byte-exact (it targets the RFC 9562 wire format, not PHP).

Why HybridId?

| Feature | HybridId | TypeID | KSUID | UUIDv7 | NanoID | CUID2 | |---|---|---|---|---|---|---| | Length | 16–24 chars | 26 chars | 27 chars | 36 chars | 21 chars | 24 chars | | Configurable size | Yes | No | No | No | No | No | | Type prefixes | Yes | Yes | No | No | No | No | | Time-sortable | Yes | Yes | Yes | Yes | No | No | | Metadata extraction | Full | Partial | Partial | Partial | None | None | | Zero dependencies | Yes | Varies | Varies | Yes | Varies | Varies | | Range queries | Yes | No | No | No | No | No | | Multi-node safe | Yes | Yes | No | Yes | N/A | N/A | | Random entropy | 47.6 – 83.4+ bits | ~80 bits | 128 bits | 74 bits | ~126 bits | ~120 bits |

Installation

npm install hybrid-id

Requires Node.js ≥ 22. No runtime dependencies. Ships ESM + CJS with bundled .d.ts types.

Quick Start

import { HybridIdGenerator } from 'hybrid-id';

const gen = new HybridIdGenerator({ node: 'A1' });

gen.generate();        // '0VBFDQz4A1Rtntu09sbf'
gen.generate('usr');   // 'usr_0VBFDQz4A1Rtntu09sbf'
gen.compact('log');    // 'log_0VBFDQz6xK9mLp2w'
gen.extended('txn');   // 'txn_0VBFDQz7A1pBKVwwn2xiF0'

CommonJS works too:

const { HybridIdGenerator } = require('hybrid-id');

Profiles

Three built-in profiles with different size/entropy tradeoffs:

| Profile | Length | Structure | Random entropy | Use case | |---|---|---|---|---| | compact | 16 | 8ts + 8rand | 47.6 bits | Internal PKs, low-scale apps | | standard | 20 | 8ts + 2node + 10rand | 59.5 bits | General purpose (default) | | extended | 24 | 8ts + 2node + 14rand | 83.4 bits | High-scale, public-facing IDs |

Standard / Extended:          Compact (no node):

0VBFDQz4 A1 Rtntu09sbf        0VBFDQz4 xK9mLp2w
|______| |_| |_________|      |______| |________|
   ts   node   random            ts      random
  • ts (8 chars): millisecond timestamp in base62. Enables chronological sorting.
  • node (2 chars, standard/extended): server/process identifier. Prevents cross-node collisions.
  • rand (variable): cryptographically secure random bytes via node:crypto.

Custom profiles are available via ProfileRegistry — see API Reference.

Configuration

The constructor takes a single options object:

import { HybridIdGenerator } from 'hybrid-id';

// Standard profile with explicit node (recommended for production)
const gen = new HybridIdGenerator({ node: 'A1' });

// Explicit profile
const gen = new HybridIdGenerator({ profile: 'extended', node: 'A1' });

// Compact — no node needed
const gen = new HybridIdGenerator({ profile: 'compact' });

// From environment variables
const gen = HybridIdGenerator.fromEnv();

By default, standard and extended profiles require an explicit node to prevent accidental collisions in production. Pass requireExplicitNode: false for local development.

Environment variables

HybridIdGenerator.fromEnv() reads from process.env:

| Variable | Default | Description | |---|---|---| | HYBRID_ID_PROFILE | standard | compact, standard, extended (or a registered custom profile) | | HYBRID_ID_NODE | — | 2-char base62 node identifier | | HYBRID_ID_REQUIRE_NODE | 1 | Set to 0 to disable the explicit-node guard | | HYBRID_ID_BLIND | 0 | Set to 1 to enable blind mode | | HYBRID_ID_BLIND_SECRET | — | Base64-encoded persistent HMAC secret | | HYBRID_ID_MAX_LENGTH | — | Hard cap on full ID length |

This library reads process.env directly and bundles no .env loader, so it stays dependency-free regardless of how you load configuration. Loading the .env is the app's job — same split as the PHP package, which reads the environment but leaves .env loading to phpdotenv / the framework.

  • Recommended: dotenv — the de-facto standard, works on every Node version and runner. Install it in your app and load it before constructing the generator:

    import 'dotenv/config';
    import { HybridIdGenerator } from 'hybrid-id';
    
    const gen = HybridIdGenerator.fromEnv();
  • Zero-dependency alternative: the native flag node --env-file=.env your-app.js (Node ≥ 20.6) — nothing to install.

Either way, hybrid-id itself never adds a dependency; the choice is yours.

⚠️ HYBRID_ID_NODE and HYBRID_ID_BLIND_SECRET are sensitive config — treat the secret like a credential. Never expose it through a client-bundled env (e.g. a VITE_-prefixed variable): those ship to the browser. The blind secret stays server-side only.

Prefixes

Stripe-style prefixes make IDs self-documenting:

gen.generate('usr');   // 'usr_0VBFDQz4A1Rtntu09sbf'
gen.generate('ord');   // 'ord_0VBFDQz5A1xiF0G9pBKV'

Rules: 1–8 chars, lowercase alphanumeric, starts with a letter. All extraction and validation functions handle prefixed IDs transparently.

Metadata & parsing

Metadata helpers are standalone, tree-shakeable functions — import only what you use:

import { parse, extractTimestamp, extractDate, extractNode, detectProfile } from 'hybrid-id';

extractTimestamp('0VBFDQz4A1Rtntu09sbf'); // 1739750400000 (ms since epoch)
extractDate('0VBFDQz4A1Rtntu09sbf');      // Date
extractNode('0VBFDQz4A1Rtntu09sbf');      // 'A1' (null for compact)
detectProfile('0VBFDQz4A1Rtntu09sbf');    // 'standard'

const p = parse('usr_0VBFDQz4A1Rtntu09sbf');
if (p.valid) {
  p.prefix;    // 'usr'
  p.profile;   // 'standard'
  p.timestamp; // 1739750400000
  p.node;      // 'A1'
  p.random;    // 'Rtntu09sbf'
}

parse() returns a discriminated union on valid, so TypeScript narrows the component fields for you. See the API Reference for the full surface (validation, sorting, range queries, custom profiles, the HybridId value object).

Database

Column sizing

| Profile | No prefix | With prefix (max 3) | With prefix (max 8) | |---|---|---|---| | compact | CHAR(16) | VARCHAR(20) | VARCHAR(25) | | standard | CHAR(20) | VARCHAR(24) | VARCHAR(29) | | extended | CHAR(24) | VARCHAR(28) | VARCHAR(33) |

Collation (MySQL/MariaDB)

Base62 uses mixed case (A != a). You must use ascii_bin or utf8mb4_bin collation — the default utf8mb4_0900_ai_ci will silently break uniqueness and sort order.

CREATE TABLE users (
    id CHAR(20) COLLATE ascii_bin NOT NULL PRIMARY KEY
    -- ...
);

PostgreSQL and SQLite are case-sensitive by default — no special collation needed.

Storage efficiency

| Format | Size | Savings vs UUID | |--------|------|-----------------| | UUID (canonical) | CHAR(36) | — | | ULID | CHAR(26) | 28% | | TypeID | VARCHAR(34) | 6% | | HybridId compact | CHAR(16) | 56% | | HybridId standard | CHAR(20) | 44% | | HybridId extended | CHAR(24) | 33% |

Smaller primary keys improve B-tree index density and reduce page splits. Time-sorted layout eliminates the random-insert penalty of UUID v4. See the Database Guide for time-range queries, NoSQL patterns, and migration strategies.

Security

Not for secrets. Do NOT use HybridId for security tokens, session IDs, API keys, or password resets. The timestamp is predictable — use crypto.randomBytes() with 128+ bits for those.

Standards alignment:

  • RFC 9562: UUIDv8-compliant via toUUIDv8()
  • CSPRNG: node:crypto randomBytes(), backed by OS-level cryptographic random
  • RFC 3986: URL-safe base62, no percent-encoding needed
  • Rejection sampling eliminates modulo bias in the random field

What HybridId is NOT: not a secret-bearing token, not constant-time in validation, timestamps are predictable by design (same as UUIDv7).

Blind Mode

HMAC-hashes the timestamp and node with a per-instance secret, making creation time unextractable. Same length and format — an observer cannot tell if an ID is blind.

const gen = new HybridIdGenerator({ node: 'A1', blind: true });
gen.generate('usr'); // 'usr_<opaque 20 chars>'

See Blind Mode for what works, what changes, and persistent secrets.

UUID Interoperability

Convert between HybridId and RFC 9562 UUIDs:

| Function | Lossless | Notes | |--------|----------|-------| | toUUIDv8() / fromUUIDv8() | Yes | Profile auto-detected on decode | | toUUIDv7() / fromUUIDv7() | No | Timestamp-preserving, needs profile hint | | toUUIDv4Format() / fromUUIDv4Format() | No | Lossy, NOT a true UUIDv4 |

import { toUUIDv8, fromUUIDv8 } from 'hybrid-id';

const uuid = toUUIDv8('0VBFDQz4A1Rtntu09sbf'); // RFC 9562 v8
fromUUIDv8(uuid);                              // '0VBFDQz4A1Rtntu09sbf'

Compact and standard profiles only. Prefixed IDs are rejected — strip the prefix first. See UUID Interoperability.

CLI

A hybrid-id binary ships with the package:

npx hybrid-id generate -p compact -n 10
npx hybrid-id inspect usr_0VBFDQz4A1Rtntu09sbf
npx hybrid-id profiles
npx hybrid-id generate --json -n 3

See the CLI Reference.

Testing & Dependency Injection

Type against the IdGenerator interface and inject a MockHybridIdGenerator in tests:

import { type IdGenerator, MockHybridIdGenerator } from 'hybrid-id';

const mock = new MockHybridIdGenerator(['ord_test001', 'ord_test002']);
mock.generate(); // 'ord_test001'

See Dependency Injection & Testing.

Learn More

| Topic | Link | |---|---| | Full API (validation, parsing, metadata, sorting, custom profiles, value object) | docs/api-reference.md | | UUID conversion (v8, v7, v4-format) | docs/uuid-interoperability.md | | Database (time-range queries, NoSQL, migration from UUID) | docs/database.md | | Blind mode (HMAC-hashed timestamps) | docs/blind-mode.md | | CLI reference | docs/cli.md | | Dependency injection & testing | docs/dependency-injection.md | | Internals (clock drift, concurrency, design decisions) | docs/internals.md | | Changelog | CHANGELOG.md |

License

MIT