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

@moltendb-web/core

v1.8.0

Published

MoltenDb WASM runtime — the database engine, Web Worker, and main-thread client in one package.

Readme

MoltenDb Web

🌋 The Embedded Database for the Modern Web

High-performance Rust engine compiled to WASM. Persistent storage via OPFS.

Interactive DemoCore EngineQuery BuilderOriginal RepositoryLicense

NPM Version License WASM Status


What is MoltenDb Web?

MoltenDb is a JSON document database written in Rust that runs directly in your browser. Unlike traditional browser databases limited by localStorage quotas or IndexedDB's complex API, MoltenDb leverages the Origin Private File System (OPFS) to provide a high-performance, append-only storage engine.

Beyond being a full-featured embedded database, MoltenDb can also serve as a persistent state manager for your application. Because all data is written to OPFS, your app's state survives page reloads, browser crashes, and unexpected connection loss — your users will never lose their work.

✅ Stable — The core engine, multi-tab sync, storage layer, and transparent at-rest encryption are feature-complete and stabilised. Server sync and analytics are planned for a future release.

🎮 Explore the Full Functionality

The best way to experience MoltenDb is through the Interactive Demo on StackBlitz. It provides a complete, live environment where you can test query builder expressions, perform mutations, and see real-time events with zero local setup.

Prefer to run it in your own environment? You can clone the demo repository to inspect the source code, run the explorers locally, and experiment with your own schema.

⚠️ Note for Online IDEs: If you are viewing this on StackBlitz or CodeSandbox, the WASM engine may be blocked by iframe security restrictions. Please click the "Open in New Window/Tab" button in the preview pane to enable the full OPFS storage engine.

Core Features

  • Hybrid Bitcask Storage: The same query logic used in our server binary, compiled to WebAssembly. Data is paged between RAM and OPFS to handle datasets larger than memory.
  • At-Rest Encryption: Transparently secure your data in the browser using XChaCha20-Poly1305 (Argon2id key derivation).
  • OPFS Persistence: Data persists across page reloads in a dedicated, high-speed sandbox.
  • Worker-Threaded: The database runs entirely inside a Web Worker—zero impact on your UI thread.
  • Multi-Tab Sync (stabilised): Leader election via the Web Locks API ensures only one tab owns the OPFS handle. All other tabs proxy reads and writes through a BroadcastChannel. Seamless leader promotion when the active tab closes.
  • Automatic Compaction: The engine automatically compacts the append-only log when it exceeds 500 entries or 5 MB, keeping storage lean without any manual intervention.
  • Real-Time Pub/Sub: Every write and delete emits a typed DbEvent to all open tabs instantly. The subscribe() pattern supports multiple independent listeners per tab — perfect for modern UI frameworks like React and Angular.
  • GraphQL-style Selection: Request only the fields you need (even deeply nested ones) to save memory and CPU.
  • Auto-Indexing: The engine monitors your queries and automatically creates indexes for frequently filtered fields.
  • Conflict Resolution: Incoming writes with _v ≤ stored _v are silently skipped.
  • Inline reference embedding (extends): Embed data from another collection at insert time.

Installation

MoltenDb is split into two packages: the core engine and the type-safe, chainable query builder.

# Install the core engine and WASM artifacts
npm install @moltendb-web/core

# Install the chainable query builder
npm install @moltendb-web/query

📦 Bundler Setup

MoltenDb handles its own Web Workers and WASM loading automatically. However, depending on your build tool, you may need a tiny config tweak to ensure it serves the static files correctly.

For Vite: Exclude the core package from pre-bundling in your vite.config.js:

// vite.config.js`
export default defineConfig({
  optimizeDeps: { exclude: ['@moltendb-web/core'] }
});

For Webpack 5 (Next.js, Create React App): Ensure Webpack treats the .wasm binary as a static resource in webpack.config.js:

module.exports = {
  module: {
    rules: [{ test: /\.wasm$/, type: 'asset/resource' }]
  }
};

Quick Start

  1. Initialize the Client

MoltenDb handles the Web Worker and WASM instantiation for you. TypeScript

import { MoltenDb } from '@moltendb-web/core';
import { MoltenDbClient, WorkerTransport } from '@moltendb-web/query';

const db = new MoltenDb('moltendb_demo', {
  encryptionKey: 'my-secret',    // Enable transparent at-rest encryption
  writeMode: 'sync'             // Ensure every write is flushed to OPFS
});
await db.init();

// Connect the query builder to the WASM worker
const client = new MoltenDbClient(db);

// 2. Insert and Query

// Use the @moltendb-web/query builder for a type-safe experience. 

// Insert data
await client.collection('laptops').set({
  lp1: {
    brand: "Apple",
      model: "MacBook Pro",
      price: 1999,
      in_stock: true,
      memory_id: 'mem1',
      specs: {
        cpu: {
          cores: 8,
          clock_speed: 3.5,
        },
        display: {
          refresh_hz: 60,
        }
      }
  },
  lp2: {
    brand: "Apple",
    model: "MacBook Air",
    price: 900,
    in_stock: true,
      memory_id: 'mem2',
      specs: {
      cpu: {
        cores: 4,
        clock_speed: 3.5,
      },
      display: {
        refresh_hz: 60,
      }
    }
  }
}).exec();

await client.collection('memory').set({
    mem1: { 
      capacity_gb: 16,
      type: 'DDR4',  
      speed_mhz: 4800,
      upgradeable: false  
    },
    mem2: {
      capacity_gb: 64,
      type: 'DDR5',  
      speed_mhz: 5600,
      upgradeable: true 
    },
}).exec();

// Query with field selection
const results = await client.collection('laptops')
  .get()
  .where({ brand: { $in: ["Apple", "Dell"] }, in_stock: true }) // Using $in operator
  .fields(['model', 'price']) // Only return these specific fields
  .sort([{ field: 'price', order: 'desc' }])
  .exec();

console.log(results); 
// [
//  {
//    "_key": "lp1",
//    "model": "MacBook Pro",
//    "price": 1999
//  },
//  {
//    "_key": "lp2",
//    "model": "MacBook Air",
//    "price": 900
//  }
// ]

// Powerful Query Capabilities
// GraphQL-style Field Selection

// Never over-fetch data again. Use dot-notation to extract deeply nested values.

await client.collection('laptops')
  .get()
  .fields(["brand", "specs.cpu.cores", "specs.display.refresh_hz"])
  .exec();

// Inline Joins

// Resolve relationships between collections at query time.

await client.collection('laptops')
  .get()
  .joins([{ 
    alias: 'ram', 
    from: 'memory', 
    on: 'memory_id', 
    fields: ['capacity_gb', 'type'] 
  }])
  .exec();

// Supported Query Operators

MoltenDb supports a variety of operators in the `where` clause:

| Operator | Aliases | Description |
|---|---|---|
| `$eq` | `$equals` | Exact equality |
| `$ne` | `$notEquals` | Not equal |
| `$gt` | `$greaterThan` | Greater than (numeric) |
| `$gte` | | Greater than or equal |
| `$lt` | `$lessThan` | Less than (numeric) |
| `$lte` | | Less than or equal |
| `$contains` | `$ct` | Substring check (string) or membership check (array) |
| `$in` | `$oneOf` | Field value is one of a list |
| `$nin` | `$notIn` | Field value is not in a list |
| `$or` | | At least one of the sub-conditions must match (array of where-style objects) |
| `$and` | | All sub-conditions must match (array of where-style objects) |

// Inline reference embedding (`extends`)

The `extends` key embeds data from another collection directly into the stored document at insert time — no join needed on reads.

```ts
await client.collection('laptops')
  .set({
    lp7: {
      brand: "MSI",
      model: "Titan GT77",
      price: 3299,
    }
  })
  .extends({
    ram: "memory.mem4",
    screen: "display.dsp3"
  })
  .exec();

When to use extends vs joins:

| | extends | joins | |---|---|---| | Resolved at | Insert time (once) | Query time (every request) | | Data freshness | Snapshot — may become stale | Always live | | Read cost | O(1) — data already embedded | O(1) per join per document | | Use when | Data rarely changes, fast reads matter | Data changes frequently, freshness matters |


Configuration

You can customise the database behavior by passing an options object to the MoltenDb constructor.

const db = new MoltenDb('my-app', {
  encryptionKey: 'user-secret',  // Secure at-rest storage in OPFS
  writeMode: 'sync'              // Ensure durability on every write
});
await db.init();

Options Reference

| Property | Type | Default | Description | | :--- | :--- | :--- |:-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | encryptionKey | string | undefined | At-Rest Encryption: If provided, all data written to OPFS is encrypted using XChaCha20-Poly1305. If omitted, data is stored as plain JSON. | | writeMode | 'async' \| 'sync' | 'async' | Durability vs Speed: 'async' is blazing fast (high throughput), while 'sync' ensures every write is flushed to disk before returning (safer but slower). Note: async is recommended for most web apps to avoid blocking during heavy write bursts. | | workerUrl | string \| URL | undefined | Custom path to the Web Worker script. | | maxBodySize | number | 10485760 | Payload Limit: Max body size in bytes. Prevents memory spikes from large messages. | | maxKeysPerRequest | number | 1000 | Batch Limit: Maximum number of keys allowed per JSON request. | | inMemory | boolean | false | Ephemeral Mode: Run entirely in RAM — no OPFS writes, no WAL. All data is lost when any tab refreshes or closes. Ideal for CI environments and ephemeral caches. |


Storage Architecture

How the Log Works

MoltenDb uses an append-only JSON log. Every write is a new line, ensuring your data is safe even if the tab is closed unexpectedly.

  • Automatic Compaction: When the log exceeds 500 entries or 5 MB, the engine automatically "squashes" the log, removing superseded document versions to reclaim space. No manual compact() calls are needed in normal operation.
  • Persistence: All data is stored in the Origin Private File System (OPFS). This is a special file system for web apps that provides much higher performance than IndexedDB.

Multi-Tab Sync

MoltenDb uses the Web Locks API for leader election. The first tab to acquire the lock becomes the leader and owns the OPFS file handle directly. Every subsequent tab becomes a follower and proxies all reads and writes through a BroadcastChannel to the leader.

When the leader tab is closed, the next queued follower automatically acquires the lock and promotes itself to leader — no data loss, no manual reconnection required.

Tab 1 (Leader) ──owns──▶ Web Worker ──▶ WASM Engine ──▶ OPFS
     │
     └── BroadcastChannel ──▶ Tab 2 (Follower)
                          ──▶ Tab 3 (Follower)

Real-Time Events (Pub/Sub)

MoltenDb has a built-in pub/sub system that automatically notifies all open tabs whenever a document is created, updated, or deleted — no polling required.

You can attach multiple independent listeners using the subscribe() method, making it trivial to keep different UI components (like React hooks or Angular signals) in sync without memory leaks:

const db = new MoltenDb('my-app');
await db.init();

// Attach a listener (Returns an unsubscribe function)
const unsubscribe = db.subscribe((event) => {
  console.log(event.event);      // 'change' | 'delete' | 'drop'
  console.log(event.collection); // e.g. 'laptops'
  console.log(event.key);        // e.g. 'lp1'
  console.log(event.new_v);      // new version number, or null on delete
});

// Later, when the UI component unmounts:
unsubscribe();

The event fires on the leader tab (directly from the WASM engine) and is automatically broadcast over the BroadcastChannel so every follower tab receives it too. This makes it trivial to keep your UI in sync across tabs without any extra infrastructure:

db.subscribe(({ event, collection, key }) => {
  if (collection === 'laptops') {
    refreshLaptopList(); // re-query and re-render
  }
});

The DbEvent type is exported from the package for full TypeScript support:

import { MoltenDb, DbEvent } from '@moltendb-web/core';

const db = new MoltenDb('my-app');
await db.init();

db.subscribe((e: DbEvent) => { /* fully typed */ });```

---

### Performance Note

Because MoltenDb uses OPFS, your browser must support `SharedArrayBuffer`. Most modern browsers support this, but your server must send the following headers:

```http
Cross-Origin-Opener-Policy: same-origin
Cross-Origin-Embedder-Policy: require-corp

Testing

The core package ships with a comprehensive test suite built on Vitest:

cd packages/core
npm test              # run all unit & integration tests
npm run test:coverage # with coverage report

What's covered

| Suite | Tests | What it verifies | |---|---|---| | init() | 5 | Leader election, idempotency, worker error propagation | | CRUD — leader | 9 | set/get/delete/getAll round-trips, collection isolation | | CRUD — follower | 3 | BroadcastChannel proxy path for all mutations | | Worker error handling | 3 | Transient errors, unknown actions, request isolation | | Leader promotion | 2 | Follower takes over when leader tab closes | | Pub/Sub (subscribe) | 2 | Multi-subscriber event delivery across tabs | | Follower timeout | 1 | Pending requests reject after 10 s if leader disappears | | terminate / disconnect | 3 | Worker cleanup, timer teardown | | Stress — rapid writes | 3 | 100 sequential, 50 concurrent, interleaved set/delete | | BC name isolation | 2 | Two databases on the same origin don't bleed data | | Bulk insert stress | 3 | 1 000 concurrent sets, 500 mixed ops, compact under pressure | | Multi-tab parallel stress | 4 | 3 tabs × 100 writes, ID collision safety, follower reads after burst, promotion under load |

Total: 50 tests — all green.


Project Structure

This monorepo contains the following packages:

  • packages/core: The core WASM engine, Web Worker logic, and the MoltenDb main client.
  • packages/query: The type-safe, chainable Query Builder.

Roadmap

  • [x] Multi-Tab Sync: Leader election for multiple tabs to share a single OPFS instance.
  • [x] Automatic Compaction: Log compacts automatically at 500 entries or 5 MB.
  • [x] Rich Test Suite: 50 unit, integration, and stress tests via Vitest.
  • [ ] React Adapter: Official @moltendb-web/react package with useQuery hooks and real-time context providers.
  • [x] Angular Adapter: Official @moltendb-web/angular package featuring Signal-based data fetching.
  • [ ] Delta Sync: Automatic two-way sync with the MoltenDb Rust server.
  • [x] Data Encryption: Transparent encryption-at-rest using hardware-backed keys (Argon2id + XChaCha20).
  • [x] Hybrid Bitcask: Seamlessly handle datasets larger than RAM by paging docs to OPFS.
  • [ ] Analytics Functionality: Run complex analytics queries straight in the browser without blocking the UI.
  • [x] Configurable Limits: User-defined RAM thresholds and request body sizes for edge and browser environments.

Contributing & Feedback

Found a bug or have a feature request? Please open an issue on the GitHub issue tracker.


License

The MoltenDb Web packages (@moltendb-web/core and @moltendb-web/query) are licensed under the MIT License.

The MoltenDb Server (Rust backend) remains under the Business Source License 1.1 (Free for organizations under $5M revenue, requires a license for managed services).

For commercial licensing or questions: [email protected]