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 🙏

© 2025 – Pkg Stats / Ryan Hefner

safelink

v0.11.3

Published

SafeLink is an open-source NodeJS library created to maintain long-term communication between distant remote sites with varying network quality

Downloads

126

Readme

SafeLink Bitdeli Badge

SafeLink is an open-source NodeJS library created to maintain long-term communication between distant remote sites with varying network quality. After deploying many different solutions for one of our customers, without much success, we decided to create our own library to satisfy our particular specifications.

Requirements

  1. No active connection. Communication should be made through basic HTTP connection without Keep-Alive. We must establish the communication channel each time to avoid having to manage connection state, reconnects, etc.
  2. Heartbeat with payload and admin response Agents must send a frequent heartbeat in order to indicate that they are still alive. The heartbeat may contain a payload containing details status information for the agent. The heartbeat response might contain administrative commands that are related to the communication itself and not application specific. For example, an admin-command might instruct the agent to send heartbeat to a different URL starting from now.
  3. Command/Response pattern The protocol must provide a way for an agent to send a response to a specific command. This bi-directional communication is critical to the usefulness of the protocol.
  4. Event Pattern An agent might subscribe to a number of network events that will be broadcast by the central process. Events are the way agents may communicate between themselves to understand the overall network condition or to react to specific events.
  5. JSON based The protocol should transport JSON payloads.

Key Features

The current version provides these key features:

  • remote command execution on the server
  • remote command execution on another agent
  • agent automatic reconnection, even after long network failures
  • distributed event handling between all connected agents and server components
  • support http polling at different intervals
  • remote command execution progress. Execute a command on any remote agents and get remote progress notifications through standard promise interface
  • promise-based remoting. No http or web socket knowledge required. Everything is hidden behind an opaque promise for the expected result.

Version History

  • 0.9.5 : Complete support for promise-based progress notification for remote commands (agent-to-agent or agent-to-dispatcher).
  • 0.3.0 : Add support for agent-connected event from the dispatcher. This will help host add logic on agent connection

Getting Started

Installation

You install Safelink in your project using NPM

npm install safelink --save

Setting up a dispatcher

Basic HTTP server

SafeLink dispatcher is the server side component that is responsible for mamaging agent communication links. All communication messages are transported through HTTP and the simplest way to use Safelink is to instantiate the dispatcher and start listening on a given port (default to 9090).

var Dispatcher = require('safelink').Dispatcher;

var dispatcher = new Dispatcher({
    port:9090
});

dispatcher.listen().then(function() {
    dispatcher.log.info("Listening for incoming agent connections");
}, function(err) {
	dispatcher.log.error(err, "General communication error");
});

Everywhere in the library, we use promises from the Q library. The promise abstraction is better than simple callbacks, even more for remoting software where responses may or may not be available directly (or cached). The power of Q promises is that it will return the result if the promise has already been fulfilled, thus freeing the client from knowing the actual state of the request.

Adding server-side commands support

In order to extend your dispatcher semantic, you may register a number of command handlers. These handlers will be executed on the server when specific command keys are executed by any connected agents.

dispatcher.commandHandlers['my-basic-command'] = function(cmd, deferredResult) {
    if(cmd.payload.myParam) {
        // do something
        deferredResult.resolve({ok:1});
    }
    else
        deferredResult.reject({ok:0});
}

This simple command handler will be triggered each time a agent perform the following :

agent.execute('my-basic-command', {myParam:'foo'}).then(function(result) {
    // Handle the successful result
}, function(err) {
    // Handle the error result
});