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

mongo-restifier

v2.5.1

Published

A restful apification library for MongoDB with build-in oAuth2 based authentication and authorization

Downloads

46

Readme

mongo-restifier

Easy to use RESTful API for MongoDB with build-in OAuth2 implementation.

Features

Installation

This module is installed via npm:

$ npm install mongo-restifier --save

Versions

Check change log to know more about versions.

Get Started

The following example serves the Todo model on a RESTful API.

  1. Install MongoDB and startup
$ mongod --syslog --fork
  1. Create npm project
$ mkdir server
$ cd server
$ npm init
  1. Install mongo-restifier
$ npm install mongo-restifier --save
  1. Create file api.js and copy the following code
// import
var mongoRestifier = require('mongo-restifier');

// configure the api
mongoRestifier('./api.conf.json')

// define "Todo" model
.registerModel({

    // mandatory
    name: 'Todo',

    // api end point (optional)
    url: '/todo',

    // schema definition - supports everything that mongoose schema supports
    schema: {
      index: {
        type: Number,         // type of this attribute
        autoIncrement: true,  // auto increment this attribute
        idField: true         // serves as id attribute replacing _id
      },
      title: {
        type: String,
        required: true
      },
      description: String,    // attribute definition can be as simple as this
      status: String,
    }

})

// ... more models can be added here

// and finally startup the server
.startup();
  1. Create configuration file api.conf.json
{
    "database": {
        "url": "mongodb://localhost/dbname"
    }
}
  1. And finally run
$ node api.js

REST API

Query

GET /todo HTTP/1.1
GET /todo/{id} HTTP/1.1
GET /todo?{field}={value} HTTP/1.1
GET /todo?fields={field1},-{field2} HTTP/1.1
GET /todo?sort={field1},-{field2} HTTP/1.1
GET /todo?count=true HTTP/1.1
GET /todo?limit=10&skip=10 HTTP/1.1

Querying takes in the following parameters:

| Parameter | Purpose | --------- | :-------------------------------- | field | Replace field with any field in your Mongoose model, and it will check for equality. | fields | Comma-delimited list of fields to populate or field names with - sign at the beginning, to omit. | sort | Sorts by the given fields in the given order, comma delimited. A - sign will sort descending. | limit | Limits the number of returned results. All results are limited to 100 by default. | skip | Skips a number of results. Useful for pagination when combined with limit. | count | Set count = true to get the count of matching items instead of items themselves.

Advanced Query

POST /todo HTTP/1.1

{title: "Your title", "status": "new" }
POST /todo HTTP/1.1

{ "status": { "$in": ["new", "hold"] }}
POST /todo HTTP/1.1

{"index": { "$gte": 1, "$lte": 100 }}
POST /todo HTTP/1.1

{ "$or" : [{ "index": 100 }, { "index": 101 }] }
POST /todo HTTP/1.1

{ "$and": [{ "index": { "$ne": 100 }}, { "index": { "$lt": 120 }}] }
POST /todo HTTP/1.1

{ "title": { "$regex": "^S.mple.*"} }
POST /todo HTTP/1.1

{ "index": { "$exists": true }}

Use POST to perform advanced queries. Query parameters remain same as of GET. Additionally the BODY of the request we can contain:

Comparison Operators

| Operator | Purpose | Example | --------- |:---------------------------| --------------------------------- | $gt | Greater than | { "age": { "$gt": 10 } } | $gte | Greater than or equal | { "age": { "$gte": 10 } } | $lt | Less than | { "age": { "$lt": 10 } } | $lte | Less than or equal | { "age": { "$lte": 10 } } | $ne | Not equal | { "status": { "$ne": "new" } } | $in | In operator | { "status": { "$in": ["new", "hold"] } } | $nin | Not In operator | { "status": { "$nin": ["new", "hold"] } } | $all | All in an array match | { "status": { "$all": ["new", "hold"] } } | $regex | Regular expression match | { "title": { "$regex": "^S.mple.*"} } | $exists | Check if a field exists | { "index": { "$exists": true }}

Logical Operators

| Operator | Purpose | Example | --------- |:---------------------| --------------------------------- | $and | Logical AND | { "$and": [{ "index": 100 }, { "status": "new" }] } | $or | Logical OR | { "$or": [{ "index": 100 }, { "index": 101 }] }

Create or Update

PUT /todo HTTP/1.1

{ "title": "Your title", "status": "new" }
PUT /todo/{id} HTTP/1.1

{ "title": "Your title", "status": "new" }
PUT /todo HTTP/1.1

{ "id": "abc", "title": "Your title" }
PUT /todo?createOnly=true HTTP/1.1

{ "id": "abc", "title": "Your title" }
PUT /todo?updateOnly=true HTTP/1.1

{ "id": "abc", "title": "Your title" }

| Parameter | Purpose | --------------- |:--------------------- | createOnly | Create an item if it doesn't exist, ignore updates | updateOnly | Update an item if it exists, ignore creates

Bulk Create or Update

PUT /todo HTTP/1.1

[{ "title": "Your title"}, { "title": "Your title"}]
PUT /todo HTTP/1.1

[{ "id": "2sd233", "title": "Sample"}, { "id": "2sd234", "title": "Sample"}]
PUT /todo?createOnly=true HTTP/1.1

[{ "id": "2sd233", "title": "Your title"}]
PUT /todo?updateOnly=true HTTP/1.1

[{ "id": "2sd233", "title": "Your title"}]

NOTE: createOnly and updateOnly are applicable for bulk updates as well. For all create and update operations, existance of an item is determinded through it's id

Delete

DELETE /todo/{id} HTTP/1.1
DELETE /todo HTTP/1.1

{ "status": "new" }
DELETE /todo HTTP/1.1

NOTE: DELETE /todo deletes everything. USE WITH CATION

Manage History

If you set history: true for a schema each change will be logged into _history table. The following apis can be used to manage versions. Please note version starts with 0. The highest version will always be the current version.

List all versions:

GET /todo/{id}/version HTTP/1.1

Get a specific version:

GET /todo/{id}/version/{version_number} HTTP/1.1

Delete a specific version:

DELETE /todo/{id}/version/{version_number} HTTP/1.1

Rollback to a specific version:

POST /todo/{id}/rollback/{version} HTTP/1.1

Rollback will delete all greater versions. Eg: if you give rollback to version 1, api will delete all entries with version greater than 1.

Model Configuration

Configuration

* - are mandatory

| Option | Type | Purpose | ---------------- | :--------- | :------------------------------------------------ | name * | Definition | Name of the model which will also be used as collection's name. Therefor no special characters (including space) other than _ is allowed. | schema * | Definition | Define the schema as defined by mongoose. See Schema Definition for details. | url | Definition | Serving endpoint. If not defined, this will be derived from name. Eg: name Todo will result in url /todo. The final url will be http://{app}:{port}/{baseUrl}/{url} | projection | Behavior | Coma separated list of fields that needs to be projected. Use - at the beginning of the fieldname to hide it. Usage: projection: 'userId,name,roles,-password' | history | Behavior | If set to true history of records will be kept in a separate collection with name <name>_history. /version apis can be used to manage versions. | userSpace | Behavior | Keep track of the user for each record and restrict access to the corresponding users. api.oauth2.enable must be true to use this option, otherwise the startup will fail with an error message. Usage: userSpace: true or userSpace: {field: "_user", ignore: ["role1"]} | timestamps | Behavior | If set to true, two fields createdAt and updatedAt are added to the schema and maintained by the api. | strict | Behavior | Set false to add any arbitral field. By default this is set to true. | configure | Function | Use this function to register middleware or plugins. Context of this function will contain the second parameter of 'defineModel' (this object itself) this.schema - Schema defined by schema, *this.model* - reference to [mongoose.Model](http://mongoosejs.com/docs/models.html), *this.modelSchema* - reference to [mongoose.Schema](http://mongoosejs.com/docs/guide.html) | beforeSave | Function | If defined, this function will be called before saving an item. This can be used to perform any calculation on the records. The signature of the function is function (item, request) { return item }` - 'item' will be an array if there are multiple items.

...
// define "Todo" model
.registerModel({

    // mandatory
    name: 'Todo',

    /** schema definition (mandatory) **/
    schema: {

        /** field definition **/
        field1: {

            /** type of the field **/
            type: String | Number | Date,

            /** validations **/
            required: true,                      // check for null values
            required: [true, 'phone# required'], // custom message
            enum: ['Coffee', 'Tea'],             // only if type = String
            minlength: Number,                   // only if type = String
            maxlength: Number,                   // only if type = String
            min: Number,                         // only if type = Number
            max: Number,                         // only if type = Number

            /** custom validation **/
            validate: {
                validator: function(v) {
                    return /\d{3}-\d{3}-\d{4}/.test(v);
                },
                message: '{VALUE} is not a valid phone number!'
            },

            /** other options **/
            idField: Boolean,           // replaces _id with field1. one and only one field can be id
            autoIncrement: Boolean,     // only if type = Number
            startAt: Number,            // only if type = Number and autoIncrement = true
            incrementBy: Number         // only if type = Number and autoIncrement = true
        }
    },

    url: String,                   // service endpoint eg: /todo
    userSpace: Boolean,            // true | { field: "{your field name}", ignore: ["role1"] } only if api.oauth2.enable = true
    strict: Boolean,               // true (default value). Set false to add any arbitral field

    configure: function() {
        // write your code here to customize this model further

        // this.schema         - The second parameter of 'defineModel' (this object itself)
        // this.model          - reference to mongoose.Model
        // this.modelSchema    - reference to mongoose.Schema
    },

    beforeSave: function(item) {
      // write your logic here to derive, filter out, add items
      // 'item' will be array if there are multiple items
      return item
    }

})
...

Schema Definition

* - are mandatory ** - additional feature than the ones supported by mongoose

| Option | Type | Purpose | ------------------------- |:---------- | :------------------------------------------------ | {field}.type * | Definition | Valid values are String, Number and Date | {field}.idField **| Behaviour | If set to true, system replaces _id with the given field. One and only one field can be set as id field. Usage: idField: true | {field}.autoIncrement **| Behaviour | If set to true, system increments value of the given field for every insertion. This validation can be applied only on Number fields . Usage: autoIncrement: true | {field}.startAt **| Behaviour | Start autoIcrement at the given value. This configuration takes effect only if autoIcrement is set to true. Usage: startAt: 1 | {field}.incrementBy **| Behaviour | Start autoIcrement with the given steps. This configuration takes effect only if autoIcrement is set to true. Usage: incrementBy: 1 | {field}.default | Behaviour | Sets the given value as the default value of the field if not specified in the request. Usage: default: Date.now | {field}.required | Validation | If set to true adds a required validator. If a value is not specified while creating an entity, operation will be rejected with an error, unless 'default' value is configured. Required can be configured as:. Usage: required: true or required: [true, 'User phone number required'] This format is applicable to all validators except validate | {field}.enum | Validation | Validates the value of a field against predefined enum values; This validation can be applied only on String fields. Usage: enum: ['Coffee', 'Tea'] | {field}.minlength | Validation | Validates the length of the value to be minimum of given value; This validation can be applied only on String fields. Usage: minlength: 5 | {field}.maxlength | Validation | Validates the length of the value to be maximum of given value; This validation can be applied only on String fields. Usage: maxlength: 5 | {field}.min | Validation | Validates the value to be minimum of given value; This validation can be applied only on Number fields. Usage: min: 5 | {field}.max | Validation | Validates the value to be maximum of given value; This validation can be applied only on Number fields. Usage: max: 5 | {field}.validate | Validation | Helps in defining custom validation. validate: { validator: function(v) { return /\d{3}-\d{3}-\d{4}/.test(v); }, message: '{VALUE} is not a valid phone number!'}

Any other additional option supported by mongoose schema

OAuth2 API

This module contains OAuth2 services configured out-of-box for password and refresh_token grant types. In this section we will discuss how to use them.

Add the following code in configuration json to enable oauth:

{
    "api": {
        "oauth2": {
            "enable": true
        }
    }
}

To know more about how to configure see API Configuration section.

Using the password grant type

First you must create a client and a user. A default client and user are configured at startup (see API Configuration for details). Additionally users and clients can be created using build in apis /oauth2/user and /oauth2/client.

To obtain a token you should POST to /oauth2/token. You should include your client credentials in the Authorization header ("Basic " + client_id:client_secret base64'd), and then grant_type (password), username and password in the request body, for example:

POST /oauth/token HTTP/1.1
Authorization: Basic czZCaGRSa3F0MzpnWDFmQmF0M2JW
Content-Type: application/x-www-form-urlencoded

grant_type=password&username=johndoe&password=A3ddj3w

Provided there weren't any errors, this will return the following (excluding the refresh_token if you've not enabled the refresh_token grant type):

HTTP/1.1 200 OK
Content-Type: application/json;charset=UTF-8

{
  "access_token":"2YotnFZFEjr1zCsicMWpAA",
  "token_type":"bearer",
  "expires_in":3600,
  "refresh_token":"tGzv3JOkF0XG5Qx2TlKWIA"
}

Using the refresh_token grant type

Send a request with previously obtained refresh_token to extend grant and expect the same output as the previous call.

POST /oauth/token HTTP/1.1
Authorization: Basic {access_token}
Content-Type: application/x-www-form-urlencoded

grant_type=refresh_token&refresh_token=tGzv3JOkF0XG5Qx2TlKWIA

Managing Users

Get

GET /oauth2/user/{userId} HTTP/1.1
Authorization: Basic base64{clientId:clientSecret}

Register User

POST /oauth2/register HTTP/1.1
Authorization: Basic base64{clientId:clientSecret}

{ "userId": "emailid", "password": "base64(plaintext)", "name": "Full name" }

Create User

PUT /oauth2/user HTTP/1.1
Authorization: Bearer {access_token}

{ "userId": "emailid", "password": "base64(plaintext)", "name": "Full name" }

Bulk Create Or Update

PUT /oauth2/user HTTP/1.1
Authorization: Bearer {access_token}

[{ "userId": "[email protected]", "password": "base64(plaintext)", "name": "Full name" },
 { "userId": "[email protected]", "password": "base64(plaintext)", "name": "Full name" }]

Delete

DELETE /oauth2/user/{userId} HTTP/1.1
Authorization: Bearer {access_token}

Delete every user in the system. NOTE: USE WITH CATION

DELETE /oauth2/user HTTP/1.1
Authorization: Bearer {access_token}

Managing Clients

Get

GET /oauth2/client/{clientId} HTTP/1.1
Authorization: Basic base64{clientId:clientSecret}

Create Client

PUT /oauth2/client HTTP/1.1
Authorization: Basic base64{clientId:clientSecret}

{ "clientId": "24x43f-sd23dde-sda23", "clientSecret": "a0c7b741-b18b-47eb", "name": "Full name" }

Bulk Create Or Update

PUT /oauth2/client HTTP/1.1
Authorization: Bearer {access_token}

[{ "clientId": "4x33-23cs34d-3ss12", "clientSecret": "a0c7b741-b18b-47eb", "name": "Full name" },
 { "clientId": "243f-sd23dde-sda23", "clientSecret": "a0c7b741-b18b-47eb", "name": "Full name" }]

Delete

DELETE /oauth2/client/{clientId} HTTP/1.1
Authorization: Bearer {access_token}

Delete every client in the system. NOTE: USE WITH CATION

DELETE /oauth2/client HTTP/1.1
Authorization: Bearer {access_token}

API Configuration

Options

All values except database.url are predefined. Specify any value in app.conf.json or app.conf.properties only if you need to override them.

| Option | Purpose | --------------------------|:------------------------------------------------------- | database.url | Mongo DB url. Format: mongodb://localhost/{dbname} | api.port | Port your server should start at; default: 5858 | api.baseUrl | Port and baseUrl defines your api url: http://{host}:{port}/{baseUrl}/ | api.cors.enable | true if you want to make cross-origin HTTP request. CORS is enabled by default. | api.cors.allowed | origin - coma separated domain names; * - for any domain, methods - coma separated values of: GET, POST, HEAD, POST, PUT , DELETE, headers - all possible headers added by user or user agent | api.oauth2.enable | true to enable OAuth2 based authentication and authorization. (false by default) | api.oauth2.default.user | To create a default user at startup, provide user attributes: name - Full name of the user; default: Superuser, userId - Required for login; default: [email protected], password - Password must be base64 encode; default: sysadmin (TODO: use encryption) , roles - the roles must be an array of string; default: [ADMIN] | api.oauth2.default.client | To create a default client at startup, provide client attributes: name, description, id, secret, grantTypes - valid values are password and refresh_token | api.oauth2.rules | Array of tab separated values in the order: AuthType - valid values are None, Basic and Bearer, Roles - coma separated values without space. eg: ADMIN,USER, Methods - coma separated values without space. eg: GET,POST,PUT, Url - eg: /api/user/**/* | api.environment | development or production | static.root | Provide a folder from the root of the project to be served as static content. By default this is set to public | static.fallback | The given file will be served if a static content is not found. Default value is index.html. Set to false to disable this feature. | logger.level | Valid values are OFF, FATAL, ERROR, WARN, LOG, INFO, DEBUG, TRACE and ALL | logger.log4js | Log4j configuration

Update properties at startup

You can optionally pass a function as second argument to mongoRestifier function inorder to modify the properties at run time.

mongoRestifier('./api.conf.json', function (properties) {
  properties.api.port = process.env.port || 3000
  return properties
})

Do more with app

Optionally you can pass on a callback function to startup function in order to use app and properties do things on your own.

mongoRestifier(...)

  .registerModel(...)
  ...

  .startup(function (app, properties, mongoose) {
    // your code
  })

Configuration File

{
  "database": {
    "url": "mongodb://localhost/{{DBNAME}}"
  },
  "api": {
    "port": 5858,
    "baseUrl": "/api",
    "environment": "production",
    "cors": {
      "enable": true,
      "allowed": {
        "origin": "*",
        "methods": "GET,PUT,POST,DELETE,OPTIONS",
        "headers": "Origin, X-Requested-With, Content-Type, Accept, Authorization, Cache-Control"
      }
    },
    "oauth2": {
      "enable": true,
      "default": {
        "user": {
          "name": "System Admin",
          "userId": "admin",
          "password": "admin",
          "roles": [
            "admin"
          ]
        },
        "client": {
          "name": "Master",
          "description": "Master Client",
          "id": "7d65d9b6-5cae-4db7-b19d-56cbdd25eaab",
          "secret": "a0c7b741-b18b-47eb-b6df-48a0bd3cde2e",
          "grantTypes": [
            "password",
            "refresh_token"
          ]
        }
      },
      "rules": [
        "None        *               OPTIONS             /api/oauth2/register",
        "None        *               OPTIONS             /api/oauth2/client",
        "None        *               OPTIONS             /api/oauth2/user",
        "Basic       *               GET                 /api/oauth2/user",
        "Bearer      ADMIN           *                   /api/oauth2/user",
        "Bearer      ADMIN           *                   /api/oauth2/client"
      ]
    }
  },
  "static": {
    "root": "public",
    "fallback": "index.html"
  },
  "logger": {
    "level": "INFO",
    "log4j": {
      "appenders": [{
        "type": "console"
      }]
    }
  }
}

ABOUT

Angular2 Integration

This software was built with Angular 2 in mind. Client-side library is being build. Come back later.

Sample Code

Check demo/api.js and demo/api.conf.json to understand the usage. npm start on this repository will run the api.

Running the Tests

Do npm install to install all of the dependencies, ensure that MongoDB is installed and running, then to run unit tests use:

$ npm test

Contributing

Contributions are very welcome! Just send a pull request. Feel free to contact me or checkout my Github page.

Author

Rinto Jose (rintoj)

Follow me: Github | Facebook | Twitter | Google+ | Youtube

License

The MIT License (MIT)

Copyright (c) 2016 Rinto Jose (rintoj)

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.