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

halson

v3.2.0

Published

The HAL+JSON Resource Object

Readme

HALSON

Build Status

The HAL+JSON Resource Object with TypeScript support.

Version

3.2.0

Installation

node.js

npm install halson --save

Bower

bower install halson --save

Example

var halson = require('halson');

var embed = halson({
        title: "joyent / node",
        description: "evented I/O for v8 javascript"
    })
    .addLink('self', '/joyent/node')
    .addLink('author', {
        href: '/joyent',
        title: 'Joyent'
    });

var resource = halson({
        title: "john doe",
        username: "doe",
        emails: [
            "[email protected]",
            "[email protected]"
        ]
    })
    .addLink('self', '/doe')
    .addEmbed('starred', embed);

console.log(resource.title);
console.log(resource.emails[0]);
console.log(resource.getLink('self'));
console.log(resource.getEmbed('starred'));
console.log(JSON.stringify(resource));

TypeScript Support

HALSON supports TypeScript with generic typing for strongly-typed HAL+JSON resources. This allows you to work with your domain objects while preserving all HAL+JSON functionality.

Basic TypeScript Usage

import halson from "halson";

// Define your domain interface
interface User {
  id: number;
  name: string;
  email: string;
  isActive: boolean;
}

// Create a typed resource
const userData: User = {
  id: 1,
  name: "John Doe",
  email: "[email protected]",
  isActive: true
};

// halson<T>() returns HALSONResource<T> which includes both
// your domain properties AND all HAL+JSON methods
const userResource = halson<User>(userData)
  .addLink('self', '/users/1')
  .addLink('edit', '/users/1/edit');

// ✅ Fully typed access to your domain properties
console.log(userResource.name);     // string
console.log(userResource.isActive); // boolean

// ✅ Plus all HAL+JSON functionality
console.log(userResource.getLink('self')); // HALSONLink | undefined
console.log(userResource.listLinkRels());  // string[]

Import Patterns

Option 1: Import with namespace access

import halson from "halson";

const resource = halson<User>(userData);
type UserResource = halson.HALSONResource<User>;

Option 2: Import with wildcard

import * as halson from "halson";

const resource = halson<User>(userData);
type UserResource = halson.HALSONResource<User>;

Option 3: Type-only imports (recommended)

import halson from "halson";
import type { HALSONResource } from "halson";

const resource = halson<User>(userData);
type UserResource = HALSONResource<User>;

Working with Embedded Resources

interface Repository {
  id: number;
  name: string;
  language: string;
  stars: number;
}

interface UserWithRepos {
  id: number;
  name: string;
  email: string;
}

const userWithRepos = halson<UserWithRepos>({
  id: 1,
  name: "John Doe",
  email: "[email protected]"
})
.addLink('self', '/users/1')
.addEmbed('repositories', [
  halson<Repository>({
    id: 101,
    name: "awesome-project",
    language: "TypeScript",
    stars: 42
  }).addLink('self', '/repos/101'),
  
  halson<Repository>({
    id: 102,
    name: "another-project", 
    language: "JavaScript",
    stars: 23
  }).addLink('self', '/repos/102')
]);

// ✅ Type-safe access to embedded resources
const repos = userWithRepos.getEmbeds<Repository>('repositories');
repos.forEach(repo => {
  console.log(`${repo.name} (${repo.language}) - ${repo.stars} stars`);
});

Function Signatures

// Generic function overloads
function halson(data?: string | object): HALSONResource;
function halson<T extends object>(data: string | T): HALSONResource<T>;

// Generic type alias
type HALSONResource<T extends object = object> = HALSONResource & T;

Migration from JavaScript

Existing JavaScript code works unchanged:

// This JavaScript code continues to work exactly the same
const resource = halson({
  title: "Hello World",
  count: 42
});

console.log(resource.title); // "Hello World"

When migrating to TypeScript, you can gradually add type annotations:

// Step 1: Add interface
interface Post {
  title: string;
  count: number;
}

// Step 2: Add generic type parameter
const resource = halson<Post>({
  title: "Hello World",
  count: 42
});

// Now you get full type safety!
console.log(resource.title); // ✅ TypeScript knows this is a string

Java HATEOAS-like Features

HALSON now provides enterprise-grade HATEOAS features similar to Spring HATEOAS and other Java frameworks:

Import Styles for Advanced Features

You can use either namespace access or named imports for the advanced features:

// Approach 1: Namespace access (all examples below use this)
import halson from "halson";

const resource = halson<User>(userData);
type UserResource = halson.HALSONResource<User>;
type PagedUsers = halson.PagedResource<UserList>;
const validation = resource.validate(options);
// Approach 2: Named imports (alternative style)
import halson, { 
  HALSONResource, 
  PagedResource, 
  IanaRels,
  ValidationOptions 
} from "halson";

const resource = halson<User>(userData);
type UserResource = HALSONResource<User>;
type PagedUsers = PagedResource<UserList>;
const validation = resource.validate({
  requireLinks: [IanaRels.SELF]
});

Enhanced Link Support with IANA Relations

import halson from "halson";

// Rich link objects with metadata
const resource = halson<User>(userData)
  .addLink(halson.IanaRels.SELF, {
    href: "/users/1",
    type: "application/json",
    title: "User Profile",
    method: "GET"
  })
  .addLink(halson.IanaRels.EDIT, {
    href: "/users/1",
    method: "PUT",
    type: "application/json",
    title: "Edit User"
  });

// Link existence checking
if (resource.hasLink(halson.IanaRels.EDIT)) {
  const editUrl = resource.getHref(halson.IanaRels.EDIT);
}

URI Template Support

// Add templated links
resource.addTemplate('search', '/users{?name,email,status}');

// Expand templates with variables
const searchUrl = resource.expandTemplate('search', {
  name: 'John',
  status: 'active'
});
// Result: "/users?name=John&status=active"

Navigation & Traversal

// Async navigation following links
const userOrders = await userResource.follow<Order>('orders');
const allRelated = await userResource.followAll<Resource>('related');

// URL resolution
const nextPageUrl = resource.resolve('next', { page: 2, size: 10 });

Builder Pattern

// Fluent resource construction
const resource = halson.HALResourceBuilder<User>(userData)
  .link(halson.IanaRels.SELF, '/users/1')
  .link(halson.IanaRels.EDIT, {
    href: '/users/1',
    method: 'PUT',
    type: 'application/json'
  })
  .template('search', '/users{?q}')
  .curie('acme', 'https://api.acme.com/rels/{rel}', true)
  .embed('profile', profileResource)
  .build();

Pagination Support

interface UserList {
  users: User[];
}

type PagedUsers = halson.PagedResource<UserList>;

const pagedUsers: PagedUsers = {
  users: [...],
  page: {
    number: 0,
    size: 10,
    totalElements: 100,
    totalPages: 10
  },
  // HAL methods + pagination helpers
  hasNext: () => true,
  hasPrev: () => false,
  next: () => '/users?page=1',
  prev: () => null
};

Curie Support (Compact URIs)

// Add namespace support
resource
  .addCurie('acme', 'https://api.acme.com/rels/{rel}', true)
  .addLink('acme:orders', '/users/1/orders')
  .addLink('acme:preferences', '/users/1/preferences');

// Expand compact URIs
const expandedRel = resource.expandCurie('acme:orders');
// Result: "https://api.acme.com/rels/orders"

Validation Framework

// Using namespace access
const validation = resource.validate({
  strict: true,
  requireLinks: [halson.IanaRels.SELF],
  allowMissingLinks: [halson.IanaRels.NEXT]
});

if (!validation.valid) {
  console.error('Validation errors:', validation.errors);
  console.warn('Validation warnings:', validation.warnings);
}
// Alternative: Using named imports
import halson, { IanaRels, ValidationOptions } from "halson";

const options: ValidationOptions = {
  strict: true,
  requireLinks: [IanaRels.SELF],
  allowMissingLinks: [IanaRels.NEXT]
};

const validation = resource.validate(options);

Content Negotiation

// Built-in content negotiation helpers
if (resource.accepts('application/json')) {
  const json = resource.asJson();
}

const halJson = resource.asHal();
const contentType = resource.getContentType();

Available IANA Link Relations

halson.IanaRels.SELF        // 'self'
halson.IanaRels.EDIT        // 'edit' 
halson.IanaRels.DELETE      // 'delete'
halson.IanaRels.NEXT        // 'next'
halson.IanaRels.PREV        // 'prev'
halson.IanaRels.FIRST       // 'first'
halson.IanaRels.LAST        // 'last'
halson.IanaRels.RELATED     // 'related'
halson.IanaRels.ALTERNATE   // 'alternate'
halson.IanaRels.CANONICAL   // 'canonical'
halson.IanaRels.COLLECTION  // 'collection'
halson.IanaRels.ITEM        // 'item'
// ... and more

API

halson([data])

Create a new HAL+JSON Resource Object.

  • data (optional): Initial data as serialized string or Object.
// empty HAL+JSON Resource Object
var resource = halson();

// resource from a serialized data
var resource = halson('{title:"Lorem Ipsum",_links:{self:{href:"/ipsum"}}');

// resource from an Object
resource = halson({
    _links: {
        self: {
            href: {"/ipsum"}
        }
    },
    title: "Lorem Ipsum"
});

// resource from another resource (no-op)
var resourceX = halson(resource);
console.log(resource === resourceX); // true

HALSONResource#listLinkRels()

List all link relations.

var data = {
    _links: {
        self: {href: '/doe'},
        related: [
            {href: 'http://doe.com'},
            {href: 'https://twitter.com/doe'}
        ]
    }
}

var resource = halson(data);
console.log(resource.listLinkRels()); // ['self', 'related']

HALSONResource#listEmbedRels()

List all link relations.

var data = {
    _embedded: {
        starred: {
            _links: {
                self: {href: '/joyent/node'}
            }
            title: "joyent / node",
            description: "evented I/O for v8 javascript"
        }
    }
}

var resource = halson(data);
console.log(resource.listEmbedRels()); // ['starred']

HALSONResource#getLinks(rel, [filterCallback, [begin, [end]]])

Get all links with relation rel.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of links. doc
  • begin, end (optional): slice filtered links. doc
var twitterLinks = resource.getLinks('related', function(item) {
    return item.name === "twitter";
});

HALSONResource#getLink(rel, [filterCallback, [default]])

Get first link with relation rel.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of links. doc
  • default (optional): Default value if the link does not exist.
var firstRelatedLink = resource.getLink('related');

HALSONResource#getEmbeds(rel, [filterCallback, [begin, [end]]])

Get all embedded resources with relation rel.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of embeds. doc
  • begin, end (optional): slice filtered links. doc
var embeds = resource.getEmbeds('starred');

HALSONResource#getEmbed(rel, [filterCallback, [default]])

Get first embedded resource with relation rel.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of embeds. doc
  • default (optional): Default value if the link does not exist.
var nodeProject = resource.getEmbed('starred', function(embed) {
    return embed.getLink('self', {}).href === '/joyent/node';
});

HALSONResource#addLink(rel, link)

Add a link with relation rel.

  • rel (required): Relation name.
  • link (required): Link to be added (string or Object).
resource
    .addLink('related', 'http://doe.com')
    .addLink('related', {
        href: 'https://twitter.com/doe',
        name: 'twitter'
    });

HALSONResource#addEmbed(rel, embed)

Add a nested resource with relation rel.

  • rel (required): Relation name.
  • embed (required): Resource to be embedded (Object or HALSONResource).
var embed = {
    _links: {
        self: {href: '/joyent/node'}
    },
    title: "joyent / node"
}
resource.addEmbed('starred', embed);

HALSONResource#insertEmbed(rel, index, embed)

Add a nested resource with relation rel.

  • rel (required): Relation name.
  • index (required): Index number where embed will be inserted
  • embed (required): Resource to be embedded (Object or HALSONResource).
var embed = {
    _links: {
        self: {href: '/joyent/node'}
    },
    title: "joyent / node"
};
resource.addEmbed('starred', embed); // add embed

var embed2 = {
    _links: {
        self: {href: '/joyent/node'}
    },
    title: "joyent / node"
};
resource.insertEmbed('starred', 0, embed2); // insert new embed before first item

HALSONResource#removeLinks(rel, [filterCallback])

Remove links with relation rel. If filterCallback is not defined, all links with relation rel will be removed.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of links. doc
// remove links with relation 'related' and name 'twitter'
resource.removeLinks('related', function(link) {
    return link.name === "twitter";
});

HALSONResource#removeEmbeds(rel, [filterCallback])

Remove embedded resources with relation rel. If filterCallback is not defined, all embeds with relation rel will be removed.

  • rel (required): Relation name.
  • filterCallback (optional): Function used to filter array of links. doc
// remove embedded resources with relation 'starred' and self-link '/koajs/koa'
resource.removeLinks('starred', function(embed) {
    return embed.getLink('self', {}).href === '/koajs/koa';
});

Contributors

Thank you to all contributors who have helped improve HALSON!