@hatchpad/restmon
v1.1.4
Published
NodeJS package that allows mongoose models to handle rest-like parameters and cursors.
Downloads
8
Readme
restmon
A package that implements common desired functionality of RESTful designs.
For use with Mongoose and MongoDB.
- Pagination using cursors that does not rely on page numbers or single field sorting
- Automatically duplicating String values to a lower case field for sorting
- Automatically updating an "updated" field
Installation
npm install @hatchpad/restmon --save
Usage
Schema & Model Definition
// Person.js
var mongoose = require('mongoose');
var Restmon = require('@hatchpad/restmon')(mongoose, 'a_secret_key');
var schema = {
firstName:{type:String, sortable:true}, // notice sortable property
lastName:{type:String, sortable:true},
dob:{type:Date, sortable: true}
};
var 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
var Person = require('./person.js');
var query = {};
Person.find(query)
.sort({lastName:1, firstName:-1}) // sort by lastName asc then firstName desc
.limit(2)
.exec(function(err, response) {
var personArr = response.data; // ['John Doe', 'Bill Doe']
var lastPersonCursor = response._meta.last; // see Find Response for details about the response
});
Second Page
Assume we have retained lastPersonCursor from the previous code block
var query = {};
Person.find(query)
.sort({'lastName,-firstName'}) // String representation of sort
.limit(2)
.after(lastPersonCursor) // use of "after"
.exec(function(err, response) {
var personArr = response.data; // ['Doug Dooley', 'John Smith']
});
Polling Example
First Query
// include the model object
var Person = require('./person.js');
var asOf = new Date(); // keep track of the asOf date
var query = {};
Person.find(query)
.sort({lastName:1, firstName:-1}) // sort by lastName asc then firstName desc
.limit(2)
.exec(function(err, response) {
var personArr = response.data; // ['John Doe', 'Bill Doe']
var lastPersonCursor = response._meta.last;
});
Polling
Assume we have retained lastPersonCursor and asOf from the previous code block
var 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
var personArr = response.data;
});
Updating
Example showing automatic update tracking date and sortable String duplication
var 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
var options = {ignoreCase: false};
var 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
- Boolean - whether to ignore case for Strings when sorting
- 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.
var ObjectId = ...;
var 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
};
var 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
- callback signature
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:'...', ...}, ...]
}