Skip to content

Validates JSON objects allowing you to trim specifics validations in specifics contexts or object values

Notifications You must be signed in to change notification settings

tobiadalsecco/contextual-schema-validator

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

31 Commits
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Contextual Schema Validator

EARLY ALPHA STAGE, DON'T USE IT IN PRODUCTION!

Validates JSON objects against a (JSON) schema which hallows you to trim specifics validations depending of specific contexts or object values.

Uses validator to validate data.

The idea

Writing validations can be an expensive task. Howere, there are a lots of good modules to help. But sometimes, you have to rewrite a validation for the same field several times, in differents services (or areas of an application), or even in the same piece of code. Let's take a simple product CRUD as an example: you may want the product ID to be required, but only at updating some information. This simple exception prevents you to write a single validation model. The Contextual Schema Validator allows you to set validations behaviors in a declarative and (if needed) centralized way. You can trim a certain validation only in a specific context or when the validated object has some values. You can also define which properties are to be saved and which are just 'auxiliary' data.

How to use

Example object: product

var productPayload = {
  id: 1,
  name: 'Spaghetti',
  category: 'Food',
  tags: [
    { id: 1, name: 'Italian', iShallNotBeHere: 'But i am here'},
    { id: 2, name: 'Pasta' }
  ],
  nutritionFacts: {
    sodium: 1,
    carbohydrates: 100,
    iShallNotBeHere: 'But i am here'
  },
  iShallNotBeHere: 'But i am here'
};

Writing a schema

var productSchema = {

  id: {
    checkIf: 'isInt',
    requireIt: {
      when: {
        contextIs: ['updateProduct', 'deleteProduct']
      }
    },
    defaults: {
      'addProduct': function() { return someID(); }
    }
  },

  name: {
    requireIt: {
      when: {
        contextIs: 'addProduct'
      }
    },
    checkIf: [
      'isAlpha',
      'AND',
      { 'isLength': [3, 20] }
    ],
    saveIt: 'always'
  },

  active: {
    requireIt: 'never',
    checkIf: 'isBoolean',
    defaults: {
      'addProduct': true,
      'deleteProduct': false
    }
  },

  category: {
    requireIt: {
      when: {
        contextIs: 'addProduct'
      }
    },
    checkIf: 'isAlpha',
    saveIt: {
      when: {
        contextIs: ['addProduct', 'updateProduct']
      }
    }
  },

  tags: {
    requireIt: 'never',
    type: 'array',
    itemsProperties: {
      id : {
        checkIf: 'isInt',
        saveIt: 'always'
      },
      name: {
        checkIf: 'isAlpha',
        saveIt: 'always'
      }
    },
    saveIt: 'always',
    remapIt: {
      fromArrayToMap: { 
        mapKeyField : 'id'
      }
    }
  },

  nutritionFacts: {
    requireIt: {
      when: [
        { contextIs: 'addProduct' },
        'AND',
        {
          payloadHas: {
            category: 'Food'
          }
        }
      ]
    },
    type: 'object',
    properties: {
      sodium: {
        requireIt: 'never',
        checkIf: 'isInt'
      },
      carbohydrates: {
        requireIt: 'never',
        checkIf: 'isInt'
      }
    },
    saveIt: 'always'
  }

};

Do validation

// load the module
var contextualSchemaValidator = require('contextual-schema-validator');

// set a validato instance whci will use our defined schema
var productValidator = contextualSchemaValidator.useSchema(productSchema);

// set if you want to exit on first error or validate everything
productValidator.exitOnFirstError = false; // default is true

//
productValidator.validate(
  productPayload,  // the data (JSON)
  'addProduct', // the context (string). Could be an event, a command, or just the name of the function in which this code is in
  {
    onError: function(errors){
      console.error('[ ERRORS ] ', errors);
    },
    onSuccess: function(saveData, auxData){
      console.log(' [ SAVE_DATA ] ');
      console.log(saveData); // You will probably save this data in a database or do something useful with it
      console.log(' [ AUX_DATA ] ');
      console.log(auxData); // this could be some metadata, or log data, or whatever you don't want to save
    }
  }
);

Example Result without errors

The id property will be moved to auxData. All the properties that are not in the schema will be stripped away, so the 'iShallNotBeHere' props will all go away.

// saveData
{ 
  name: 'Spaghetti',
  category: 'Food',
  tags: [ 
    { id: 1, name: 'Italian' }, 
    { id: 2, name: 'Pasta' } 
  ],
  nutritionFacts: { 
    sodium: 1, 
    carbohydrates: 100 
  }
}

// auxData
{ id: 1 }

Example result with errors

If the payload was:

var productPayload = {
  id: 1,
  name: 'Spaghetti with a very very long long name', // max is set to 20 now, will trim error
  category: 'Food',
  tags: [
    { id: 1, name: 'Italian', iShallNotBeHere: 'But i am here'},
    { /* id: 2, */ name: 'Pasta' }
  ],
  nutritionFacts: {
    sodium: 'wrong-id',
    carbohydrates: 100,
    iShallNotBeHere: 'But i am here'
  },
  iShallNotBeHere: 'But i am here'
};

then the result will be an array like:

[ 
  { type: 'invalid',  prop: 'name' },
  { type: 'required', prop: 'tags.id' },
  { type: 'invalid',  prop: 'nutritionFacts.sodium' } 
]

Documentation

Define the type of your property

You only have the 'type' property do to this. And you must set it only if the property is an object or an array.

type

Defines if the property is a simple value, an array of objects or a simple object.

Possible values:

  • 'object' : in this case you must set the 'properties' property, which accepts a nested schema object, with all the possible validations and conditions, just as the main level object.
  • 'array': in this case you must set the 'itemsProperties', which accepts an array of nested schema objects, with all the possible validations and conditions, just as the main level object.

Disabled by default, if not set the property will be treated as a simple values (string, bool or whatever).

Main validations actions

requireIt

Defines if you property will be required.

Possible values:

  • 'always'
  • 'never'
  • an object with the 'when' property

Default: 'always'.

refuseIt

Same as required, but opposite action: the properties cannot be accepted. Useful for avoiding conflicts.

Possible values:

  • 'always'
  • 'never'
  • an object with the 'when' property

Default: 'never'.

checkIf

Defines data validation.

Possible values:

  • A string with one of the validator's method name (the ones with no arguments, besides the value, which is automatically passed). Example: 'isInt'
  • An object with validator's method name as the key and an array of arguments. Example: { isLength: [3, 20] }
  • An array of multiple conditions for validation

Disabled by default.

saveIt

Defines if the property will go into the first returned object, which you will probably save in some database, or if it will go into the second object, as auxiliary data.

Possible values:

  • 'always'
  • 'never'
  • an object with the 'when' property

defaults

An object that defines the defaults values of a property depending on the context. The object pattern is:

{
  'theContext': 'the value'
}

If the property is set in the payload, even if it has a default value for the current context, the original payload value will be kept, and the default will be ignored.

sanitizeIt

Defines if the property will be sanitized by on of the validator's methods.

Possible values:

  • A string with one of the validator's method name (the ones with no arguments, besides the value, which is automatically passed). Example: 'escape'
  • An object with validator's method name as the key and an array of arguments. Example: { ltrim: [['s', '.']] }
  • An array with one of the above values as items

remapIt

Defines some aditional operation on the valid value. Accepts an object with the remap type as key and an object of configurations.

remapIt Types

Currently supports only the 'fromArrayToMap' type.

fromArrayToMap

Transform an array of objects into a map (object), with each array item mapped by the chosen item key. For example, given the value:

[
  { id: 1, name: 'Italian' }, 
  { id: 2, name: 'Pasta' } 
]

and the remaptIt configuration:

remapIt: {
  fromArrayToMap: { 
    mapKeyField : 'id'
  }
}

the value will be remapped this way:

{ 
  '1': { id: 1, name: 'Italian' },
  '2': { id: 2, name: 'Pasta' } 
}

The 'when' object

Defines if 3 of the 4 main validation (requireIt, refuseIt, saveIt) should be triggered.

Possible values:

  • An object with one Simple Condition
  • An array of Multiple Conditions

Simple Conditions

contextIs

Defines the context in which some action should take place. The context will be matched against the second argument of the main validate() function.

Possible values:

  • A string with the context name
  • An array of strings. The context should be match one of them (like an 'OR' logical operator)

contextIsNot

Same as contextIs, but opposite result.

payloadHas

Defines a condition in which if the payload (the validated object) has a property with a certain values, the action should take place.

Possible values:

  • an object with the property key as the key, and the value as the value.

Works only with main level's properties (who knows, in the future...)

Array of Multiple Conditions

The 'when' and 'checkIf' operators accept multiple conditions, as an array.

The general structure of the Multiple Conditions Array must be an alternation of Simple Conditions (or 'checkIf' values) and logical Operators, like this:

[ ConditionOrValidation, LogicalOperator, ConditionOrValidation, LogicalOperator, ...]

Logical Operators

Possible values:

  • 'AND'
  • 'OR'

About

Validates JSON objects allowing you to trim specifics validations in specifics contexts or object values

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published