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

@metreeca/qest

v0.9.2

Published

Minimalist foundations for client-driven, queryable REST/JSON APIs

Downloads

17

Readme

@metreeca/qest

npm

Minimalist foundations for client-driven, queryable REST/JSON APIs.

@metreeca/qest standardizes critical capabilities that vanilla REST/JSON APIs typically lack or implement in ad‑hoc, non‑portable ways:

  • Client-Driven: clients specify what they need, retrieving complex envelopes in a single call
  • Queryable: advanced filtering and aggregation, supporting faceted search and analytics

Developers seek these features in frameworks like GraphQL; @metreeca/qest brings them to REST/JSON, achieving:

  • Familiar Patterns: standard REST and JSON conventions, no new paradigms to learn
  • Simple Clients: no specialized libraries, preprocessors, or code generators
  • Automated Servers: model-driven development, dramatically reducing implementation effort
  • Standard Caching: compatibility with CDNs and browser caches using standard GET requests
  • URL-Based Versioning: standard REST versioning without field deprecation complexity

Installation

npm install @metreeca/qest

[!WARNING]

TypeScript consumers must use "moduleResolution": "nodenext"/"node16"/"bundler" in tsconfig.json. The legacy "node" resolver is not supported.

Usage

[!NOTE]

This section introduces essential concepts; for complete coverage, see the API reference:

| Module | Description | |----------------------------------------------------------------------------|---------------------------------| | @metreeca/qest | Shared values, types, and guards | | @metreeca/qest/state | Resource state management | | @metreeca/qest/model | Client-driven retrieval |

@metreeca/qest types define payload semantics and formats for standard REST operations:

| Method | Type | Description | |--------|--------------|------------------------------------| | GET | Resource | Resource retrieval | | GET | Resource | Collection retrieval | | GET | Model | Client-driven resource retrieval | | GET | Query | Client-driven collection retrieval | | POST | Resource | Resource creation | | PUT | Resource | Complete resource state update | | PATCH | Patch | Partial resource state update | | DELETE | IRI | Resource deletion |

Resources and Patches

A Resource is a property map describing data returned by a REST endpoint, with optional links to other endpoints:

GET https://data.example.com/products/123
{
  "id": "https://data.example.com/products/123",
  "name": "Widget",
  "category": "Electronics",
  "tags": [
    "gadget",
    "featured"
  ],
  "vendor": "https://data.example.com/vendors/456",
  "price": 99.99,
  "inStock": true
}

The same format is used for complete resource updates:

PUT https://data.example.com/products/123
({
    name: "Widget",
    category: "Electronics",
    tags: ["gadget", "premium"],
    vendor: "https://data.example.com/vendors/456",
    price: 79.99,
    // inStock                     // not included → deleted
});

A Patch describes partial updates with the same effect:

PATCH https://data.example.com/products/123
({
    tags: ["gadget", "premium"], // updated
    price: 79.99, // updated
    inStock: null, // deleted
});

Properties set to null are deleted; properties not included are unchanged.

Client-Driven Retrieval

Client-driven retrieval lets clients specify exactly what data to retrieve from both single resources and collections. Expansions and nested queries can be arbitrarily deep: no over-fetching of unwanted fields, no under-fetching requiring additional calls to resolve linked resources.

This is the core contribution of @metreeca/qest: vanilla REST/JSON APIs lack a standard way for clients to control retrieval, forcing them to accept fixed server responses or rely on ad-hoc query parameters. Client-driven retrieval fills this gap, supporting precise control over responses while remaining fully compatible with standard HTTP caching.

[!IMPORTANT]

Client-driven retrieval is fully optional. Servers may provide defaults, typically derived from the underlying data model, preserving standard REST/JSON behavior while enabling advanced capabilities when needed.

Resources — A Model defines the data retrieval envelope: which properties to include and how deeply and in how much detail to expand linked resources.

GET https://data.example.com/products/123?<model>

where <model> is the following URL-encoded JSON:

({
    id: "",
    name: "",
    price: 0,
    vendor: {
        id: "",
        name: "",
    },
});

The response includes only the requested properties, with the linked vendor expanded to show just id and name:

{
  "id": "https://data.example.com/products/123",
  "name": "Widget",
  "price": 99.99,
  "vendor": {
    "id": "https://data.example.com/vendors/145",
    "name": "Acme"
  }
}

Collections — A Query combines a projection model with filtering, ordering, and pagination criteria, also supporting computed projections including aggregates for faceted search and analytics.

GET https://data.example.com/products/?<query>

where <query> is the following URL-encoded JSON:

({
    items: [
        {
            id: "",
            name: "",
            price: 0,
            vendor: {
                id: "",
                name: "",
            },
            ">=price": 50, // filter: price ≥ 50
            "<=price": 150, // filter: price ≤ 150
            "^price": "asc", // sort: by price ascending
            "#": 25, // limit: 25 results
        },
    ],
});

A single call returns exactly what the client requested:

  • projected: product id, name, price
  • expanded: linked vendor with only id and name (not its full state)
  • filtered: price between 50 and 150
  • sorted: by price ascending
  • paginated: up to 25 results
{
  "items": [
    {
      "id": "https://data.example.com/products/456",
      "name": "Gadget",
      "price": 59.99,
      "vendor": {
        "id": "https://data.example.com/vendors/145",
        "name": "Acme"
      }
    },
    {
      "id": "https://data.example.com/products/123",
      "name": "Widget",
      "price": 99.99,
      "vendor": {
        "id": "https://data.example.com/vendors/145",
        "name": "Acme"
      }
    },
    {
      "id": "https://data.example.com/products/789",
      "name": "Gizmo",
      "price": 129.99,
      "vendor": {
        "id": "https://data.example.com/vendors/236",
        "name": "Globex"
      }
    }
  ]
}

Integrated Ecosystem

[!IMPORTANT]

@metreeca/qest defines data types only; applications are absolutely free to handle validation, storage, and publishing as they see fit.

But @metreeca/qest is also the foundation of an integrated ecosystem for rapid application development, turning those same types into a complete model-driven stack:

| Package | Description | |-----------------------------|----------------------------------------------------------------| | @metreeca/qest | Data types for client-driven, queryable REST/JSON APIs | | @metreeca/blue | Declarative blueprints for model-driven linked data processing | | @metreeca/keep (upcoming) | Shape-driven storage framework with pluggable adapters | | @metreeca/gate (upcoming) | Shape-driven REST/JSON API publishing |

JSON-LD Foundations

JSON-LD (JSON for Linked Data) is a W3C standard for publishing linked data on the web. It extends JSON with web identifiers (IRIs) to link resources across systems and domains, and to give property names precise, machine-readable meaning by mapping them to shared vocabularies — a capability at the heart of the Web Data Activity ( Semantic Web) and modern knowledge graphs.

@metreeca/qest defines a controlled JSON-LD subset designed to feel like plain idiomatic JSON, letting JavaScript developers work with linked data using familiar REST/JSON patterns without mastering JSON-LD technicalities, while retaining full compatibility with standard JSON-LD processors.

This controlled subset is specified by:

  • compacted documents with short property names and nested objects, just like regular JSON

  • ECMAScript identifiers as property names (terms), enabling dot notation access; JSON-LD keywords (@id, @type, etc.) and blank node identifiers are not allowed and must be mapped to identifiers via an application-provided @context (for instance, "id": "@id"); @context must also maps property names to IRIs for semantic interoperability

  • native JSON primitives (boolean, number, string) as values; typed literals with arbitrary datatypes are not allowed and must be represented as strings with datatype coercion declared in @context

  • language maps for localised text; @none keys for non-localised values in language maps are not allowed and must be handled using string | Local union types or the zxx language tag

  • index maps for key-indexed property values; indexed semantics must be signalled by application-provided @context declarations, as indexed values are otherwise indistinguishable from nested resources

  • IRI references for linking resources across systems and domains; data structures require absolute IRIs; codec functions handle conversion to/from root-relative forms

Support

  • open an issue to report a problem or to suggest a new feature
  • start a discussion to ask a how-to question or to share an idea

License

This project is licensed under the Apache 2.0 License – see LICENSE file for details.