1.1.5 • Published 3 years ago

@alexspirgel/schema v1.1.5

Weekly downloads
52
License
ISC
Repository
github
Last release
3 years ago

Schema

Schema is a JavaScript validator, meaning it will detect if an input matches defined constraints, it will not edit input data. This was originally developed for validating complex options objects as they are passed into other scripts, but it can be used effectively outside of that specific example.

Installation

Using NPM:

npm install @alexspirgel/schema
const Schema = require('@alexspirgel/schema');

Using a script tag:

Download the normal or minified script from the /dist folder.

<script src="path/to/schema.js"></script>

Usage

Create a schema using a model:

const schema = new Schema({
    type: 'number',
    greaterThan: 5
});

Use the schema to validate an input:

schema.validate(6); // returns true

You can specify an error style as the second parameter of the validate method. The error style options are:

  • 'throw' (default) will throw a formatted error on validate failure. The message will include multiple errors when applicable.
  • 'array' will return an array of errors on validate failure. Useful for sorting through multiple errors programatically.
  • 'boolean' will return false on validate failure.
schema.validate(4); // throws an error
schema.validate(4, 'throw'); // throws an error
schema.validate(4, 'array'); // returns an array
schema.validate(4, 'boolean'); // returns false

More complex example:

const schema = new Schema({
    required: true,
    type: 'object',
    propertySchema: {
        width: {
            required: true,
            type: 'number',
            greaterThanOrEqualTo: 0
        },
        tags: {
            type: 'array',
            itemSchema: [
                {
                    type: 'number'
                },
                {
                    type: 'string'
                }
            ]
        }
    }
});
schema.validate(null); // fail
schema.validate({}); // fail
schema.validate({ // pass
    width: 10
});
schema.validate({ // fail
    width: -5
});
schema.validate({ // pass
    width: 10,
    tags: []
});
schema.validate({ // fail
    width: 10,
    tags: ['test', true]
});
schema.validate({ // pass
    width: 10,
    tags: ['test', 123, 'hello']
});

Model Properties

These parameters are used to define the schema rule set.

This property has no restrictions on what models it can belong to.

Available values: any boolean.

Setting required to true requires an input not to be null or undefined.

Setting required to false or omitting it from the model (equivalent to undefined) will not require any input. If an input is null or undefined all other model properties will be skipped and the input is valid.

model = {
  required: true
};

This property has no restrictions on what models it can belong to.

Available values: boolean, number, string, array, object, function.

An input must match the set type.

model = {
  type: 'boolean'
};

Notes:

  • NaN is not a valid number.
  • null is not a valid object.
  • Arrays are not objects and objects are not arrays.
    • [] is not a valid object.
    • {} is not a valid array.

This property is restricted to models with a type property of boolean, number, or string.

Available values: any value or array of values.

An input must match the value or one of the values in an array of values.

model = {
  type: 'string',
  exactValue: 'hello world'
};
model = {
  type: 'number',
  exactValue: [5, 7, -12]
};

This property is restricted to models with a type property of number.

Available values: any number.

An input must be greater than the set number.

model = {
  type: 'number',
  greaterThan: 5
};

This property is restricted to models with a type property of number.

Available values: any number.

An input must be greater than or equal to the set number.

model = {
  type: 'number',
  greaterThanOrEqualTo: 5
};

This property is restricted to models with a type property of number.

Available values: any number.

An input must be less than the set number.

model = {
  type: 'number',
  lessThan: 5
};

This property is restricted to models with a type property of number.

Available values: any number.

An input must be less than or equal to the set number.

model = {
  type: 'number',
  lessThanOrEqualTo: 5
};

This property is restricted to models with a type property of number.

Available values: any number or array of numbers.

An input must be divisible by the set number or one of the numbers in the array of numbers.

model = {
  type: 'number',
  divisibleBy: 2 // even numbers
};
model = {
  type: 'number',
  divisibleBy: [5, 8]
};

This property is restricted to models with a type property of number.

Available values: any number or array of numbers.

An input must not be divisible by the set number or any of the numbers in the array of numbers.

model = {
  type: 'number',
  notDivisibleBy: 2 // odd numbers
};
model = {
  type: 'number',
  notDivisibleBy: [5, 8]
};

This property is restricted to models with a type property of string.

Available values: any number.

An input must have a character count greater than or equal to the set number.

model = {
  type: 'string',
  minimumCharacters: 5
};

This property is restricted to models with a type property of string.

Available values: any number.

An input must have a character count less than or equal to the set number.

model = {
  type: 'string',
  maximumCharacters: 5
};

This property is restricted to models with a type property of array.

Available values: any number.

An input must have length greater than or equal to the set number.

model = {
  type: 'array',
  minimumLength: 5
};

This property is restricted to models with a type property of array.

Available values: any number.

An input must have length less than or equal to the set number.

model = {
  type: 'array',
  maximumLength: 5
};

This property is restricted to models with a type property of object.

Available values: any value or array of values.

An input must be an instance of the value or one of the values in the array of values.

model = {
  type: 'object',
  instanceOf: Element
};
model = {
  type: 'object',
  instanceOf: [Element, Error]
};

This property is restricted to models with a type property of object.

Available values: any boolean.

Setting allowUnvalidatedProperties to false requires every input property to have a model defined in the propertySchema property.

Setting allowUnvalidatedProperties to true or omitting it from the model (equivalent to undefined) will not check if input properties are validated.

model = {
  type: 'object',
  allowUnvalidatedProperties: false,
  propertySchema: {
    width: {
      type: 'number'
    },
    height: {
      type: 'number'
    }
  }
};

This property is restricted to models with a type property of array or object.

Available values: a model.

Each property of the input must validate using the allPropertySchema.

model = {
  type: 'array',
  allPropertySchema: {
    type: 'number'
  }
};

This property is restricted to models with a type property of array or object.

Available values: an object containing property and model pairs.

Each property of the input object must validate using the corresponding property model defined in the model.

model = {
  type: 'object',
  propertySchema: {
    property1: {
      type: 'number'
    },
    property2: {
      type: 'string'
    }
  }
};

This property has no restrictions on what models it can belong to.

Available values: any function that returns true on successfull validation or throws a Schema.ValidationError on failure.

An input must validate successfully using the custom validation function.

model = {
  custom: (inputPathManager) => {
    if (inputPathManager.value.includes('hello')) {
      return true;
    }
    else {
      throw new Schema.ValidationError(`Custom validation failed. The input must contain the string 'hello'.`);
    }
  }
};

Multiple Models

Anywhere you could use a model, you can instead use an array of models. If an input validates successfully using any of the models in the array, the validation is successfull.

Here is an example of multiple models:

model = [
  {
    require: true,
    type: 'number'
  },
  {
    require: true,
    type: 'string'
  }
];

This model allows for the input to be either a number or a string.