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 🙏

© 2024 – Pkg Stats / Ryan Hefner

restmon

v1.1.6

Published

NodeJS package that allows mongoose models to handle rest-like parameters and cursors.

Downloads

3

Readme

restmon

A package that implements common desired functionality of RESTful designs.

For use with Mongoose and MongoDB.

  1. Pagination using cursors that does not rely on page numbers or single field sorting
  2. Automatically duplicating String values to a lower case field for sorting
  3. Automatically updating an "updated" field

Installation

npm install restmon --save

Usage

Schema & Model Definition

// Person.js
const mongoose = require('mongoose');
const Restmon = require('@hatchpad/restmon')(mongoose, 'a_secret_key');

const schema = {
  firstName:{type:String, sortable:true},   // notice sortable property
  lastName:{type:String, sortable:true},
  dob:{type:Date, sortable: true}
};

const Person = new Restmon('Person', schema);
module.exports = Person;

Queries

Let's assume the database has the following person objects:
[
{firstName: 'Bill', lastName: 'Doe',    dob:'Dec 10, 1990'},
{firstName: 'Bob',  lastName: 'Smith',  dob:'Oct 03, 1980'},
{firstName: 'John', lastName: 'Doe',    dob:'Jun 10, 1990'},
{firstName: 'Doug', lastName: 'Dooley', dob:'Mar 12, 1960'},
{firstName: 'John', lastName: 'Smith',  dob:'Jul 20, 1990'}
]

Explanation of cursors

Cursors are a String token representing all the sortable fields of an object. Using the metadata returned in a restmon query, we can easily set up pagination or polling without trusting that the data objects will not change or be removed from the database.

Pagination Example

First Page
// include the model object
const Person = require('./person.js');

const query = {};
Person.find(query)
.sort({lastName:1, firstName:-1})  // sort by lastName asc then firstName desc
.limit(2)
.exec(function(err, response) {
  const personArr = response.data;   // ['John Doe', 'Bill Doe']
  const lastPersonCursor = response._meta.last; // see Find Response for details about the response
});
Second Page

Assume we have retained lastPersonCursor from the previous code block

const query = {};
Person.find(query)
.sort({'lastName,-firstName'})    // String representation of sort
.limit(2)
.after(lastPersonCursor)          // use of "after"
.exec(function(err, response) {
  const personArr = response.data;  // ['Doug Dooley', 'John Smith']
});

Polling Example

First Query
// include the model object
const Person = require('./person.js');

const asOf = new Date();             // keep track of the asOf date

const query = {};
Person.find(query)
.sort({lastName:1, firstName:-1})  // sort by lastName asc then firstName desc
.limit(2)
.exec(function(err, response) {
  const personArr = response.data;   // ['John Doe', 'Bill Doe']
  const lastPersonCursor = response._meta.last;
});
Polling

Assume we have retained lastPersonCursor and asOf from the previous code block

const query = {};
Person.find(query)
.sort({'lastName,-firstName'})    // String representation of sort
.limit(2)
.before(lastPersonCursor)         // use of "before"
.since(asOf)
.exec(function(err, response) {
  // will only return objects that are new or have been updated
  const personArr = response.data;
});

Updating

Example showing automatic update tracking date and sortable String duplication

const person = new Person.model({
  firstName: 'Rick',
  lastName: 'Peoples'
});
Person.save(person, function(err, savedPerson) {
});

In this example, savedPerson will be this

{
  _firstName: 'rick',
  firstName: 'Rick',
  _lastName: 'peoples',
  lastName: 'Peoples',
  updated: "the current date/time"
}

Config Options

Config Options Example

const options = {ignoreCase: false};
const Restmon = require('@hatchpad/restmon')(mongoose, 'a_secret_token', options);
  • ignoreCase
    • Boolean - whether to ignore case for Strings when sorting
      • This can be overridden in the schema
    • default true
  • updated
    • String - name of the field for tracking the data
    • default 'updated'
  • id
    • String - name of the id field
    • default '_id'

Defining Restmon & Schema

A restmon schema is defined exactly the same way as a normal Mongoose schema. The only exceptions are that fields that are sortable (that need to be included in cursors) need to be marked with the sortable property and you can configure each sortable String field with ignoreCase. If a field is marked as sortable, it will be indexed, so you can exclude the index property.

const ObjectId = ...;
const schema = {
  firstName:{type:String, sortable:true},   // notice sortable property
  lastName:{type:String, sortable:true},    // sortable fields are also indexed
  nickname:{type:String, sortable:true, ignoreCase:false},    // will sort uppercase letters first
  dob:{type:Date, sortable: true},
  companyId:{type:ObjectId, index: true}     // this field is indexed but not sortable
};

const Person = new Restmon('Person', schema);
module.exports = Person;

Restmon

  • save(doc, callback)
    • Saves the doc using Mongoose.save
  • update(conditions, doc, options, callback)
    • Updates the doc using Mongoose.update
  • create(docs, callback)
    • Creates multiple docs using Mongoose.create
  • find(criteria, callback)
    • Returns a RestmonQuery object
  • findOne(criteria, callback)
    • Returns a RestmonQuery object

Remember to use the Restmon modification functions for updating and creating documents so that it will automatically track the updated date.

RestmonQuery

  • sort(sortBy)
    • sortBy can be an Ojbect like this: {firstName:1,lastName:-1,dob:1}
    • sortBy can be a String like this: 'firstName,-lastName,+dob'
    • returns RestmonQuery (this)
  • limit(limit)
    • limit - Max Number of documents to retreive
    • returns RestmonQuery (this)
  • populate(populate, select, model, match, options)
    • See Mongoose documentation
    • returns RestmonQuery (this)
  • select(select)
    • See Mongoose documentation
    • returns RestmonQuery (this)
  • after(cursor)
    • cursor - Encoded String of RestmonCursor or RestmonCursor object
    • Tells Restmon to only return documents after the cursor given the particular sort
    • return RestmonQuery (this)
  • before(cursor)
    • cursor - Encoded String of RestmonCursor or RestmonCursor object
    • Tells Restmon to only return documents before the cursor given the particular sort
    • return RestmonQuery (this)
  • since(timestamp)
    • timestamp - Date object
    • Tells Restmon to only return objects that where updated after the timestamp
    • return RestmonQuery (this)
  • until(timestamp)
    • timestamp - Date object
    • Tells Restmon to only return objects that where updated before the timestamp
    • return RestmonQuery (this)
  • exec(callback)
    • callback signature
      • function(err, result) {}
    • executes the Mongoose query

Find Response

A response for a restmon find query will have metadata plus the array of objects so it's important to document what you should expect.

Given the following code block:

Person.find({})
.limit(5)
.sort('lastName')
.exec(function(err, response) {
  ...
});

response will be similar to this:

{
  _meta: {
    first: <*cursor* of the first object in the sorted array>,
    last: <*cursor* of the last object in the sorted array>,
    freshest: <updated *Date* of the most recently updated object in the array>,
    stalest: <updated *Date* of the least recently updated object in the array>
  },
  data: [{firstName:'...', ...}, ...]
}