SimpleSchema validates JavaScript objects to ensure they match a schema. It can also clean the objects to automatically convert types, remove unsupported properties, and add automatic values such that the object is then more likely to pass validation.
There are a lot of similar packages for validating objects. These are some of the features of this package that might be good reasons to choose this one over another:
- Written in TypeScript and published with support for both CommonJS and ESM.
- Isomorphic. Works in NodeJS and modern browsers.
- Package has been maintained for 10 years and is mature
- Has nearly 500 tests and is used in production apps of various sizes
- The object you validate can be a MongoDB update document (also known as "modifier" object). SimpleSchema understands how to properly validate it such that the object in the database, after undergoing modification, will be valid.
- Object cleaning feature can fix potential validation errors automatically, avoiding unnecessary errors for your users. Cleaning also supports MongoDB update documents.
- Powerful customizable error message system with decent English language defaults and support for localization, which makes it easy to drop this package in and display the validation error messages to end users.
- Used by Mailchimp Open Commerce
- Used by the Collection2 and AutoForm Meteor packages.
There are also reasons not to choose this package. Because of all it does, this package is more complex than (but still "simple" :) ) and slower than some other packages. Based on your needs, you should decide whether these tradeoffs are acceptable. Other packages you might consider:
Table of Contents
- SimpleSchema (simpl-schema NPM package)
- Installation
- Lingo
- The History of SimpleSchema
- Quick Start
- Validate an Object and Throw an Error
- Validate an Array of Objects and Throw an Error
- Validate an Object and Get the Errors
- Validate a MongoDB Update Document
- Automatically Clean the Object Before Validating It
- Set Default Options for One Schema
- Set Default Options for All Schemas
- Explicitly Clean an Object
 
- Defining a Schema
- Schema Keys
- Schema Rules
- Validating Data
- The Object to Validate
- Ways to Perform Validation
- Validating an Object
- Validating Only Some Keys in an Object
- Validating Multiple Objects in an Array
- Validation Options
- Validating and Throwing ValidationErrors
- Custom Field Validation
- Custom Whole-Document Validators
- Manually Adding a Validation Error
- Getting a List of Invalid Keys and Validation Error Messages
 
- Customizing Validation Messages
- Other Validation Context Methods
- Other SimpleSchema Methods
- Cleaning Objects
- Dates
- Best Practice Code Examples
- Debug Mode
- Extending the Schema Options
- Converting a SimpleSchema to a JSONSchema
- Add On Packages
- Contributors
- Sponsors
- License
- Contributing
 
npm install simpl-schemaThere are other NPM packages named simpleschema and simple-schema. Make sure you install the right package. There is no "e" on "simpl".
In this documentation:
- "key", "field", and "property" generally all mean the same thing: an identifier for some part of an object that is validated by your schema. SimpleSchema uses dot notation to identify nested keys.
- "validate" means to check whether an object matches what you expect, for example, having the expected keys with the expected data types, expected string lengths, etc.
- MongoDB operators
SimpleSchema was first released as a Meteor package in mid-2013. Version 1.0 was released in September 2014. In mid-2016, version 2.0 was released as an NPM package, which can be used in Meteor, NodeJS, or static browser apps.
If you are migrating from the Meteor package, refer to the CHANGELOG
import SimpleSchema from "simpl-schema";
new SimpleSchema({
  name: String,
}).validate({
  name: 2,
});An error is thrown for the first invalid object found.
import SimpleSchema from "simpl-schema";
new SimpleSchema({
  name: String,
}).validate([{ name: "Bill" }, { name: 2 }]);import SimpleSchema from "simpl-schema";
const validationContext = new SimpleSchema({
  name: String,
}).newContext();
validationContext.validate({
  name: 2,
});
console.log(validationContext.isValid());
console.log(validationContext.validationErrors());import SimpleSchema from "simpl-schema";
const validationContext = new SimpleSchema({
  name: String,
}).newContext();
validationContext.validate(
  {
    $set: {
      name: 2,
    },
  },
  { modifier: true }
);
console.log(validationContext.isValid());
console.log(validationContext.validationErrors());TO DO
import SimpleSchema from "simpl-schema";
const mySchema = new SimpleSchema(
  {
    name: String,
  },
  {
    clean: {
      autoConvert: true,
      extendAutoValueContext: {},
      filter: false,
      getAutoValues: true,
      removeEmptyStrings: true,
      removeNullsFromArrays: false,
      trimStrings: true,
    },
    humanizeAutoLabels: false,
    requiredByDefault: true,
  }
);These options will be used every time you clean or validate with this particular SimpleSchema instance.
import SimpleSchema from "simpl-schema";
SimpleSchema.constructorOptionDefaults({
  clean: {
    filter: false,
  },
  humanizeAutoLabels: false,
});
// If you don't pass in any options, it will return the current defaults.
console.log(SimpleSchema.constructorOptionDefaults());These options will be used every time you clean or validate with any SimpleSchema instance, but can be overridden by options passed in to the constructor for a single instance.
Important notes:
- You must call SimpleSchema.constructorOptionDefaultsbefore any of your schemas are created, so put it in an entry-point file and/or at the top of your code file.
- In a large, complex project where SimpleSchema instances might be created by various JavaScript packages, there may be multiple SimpleSchemaobjects. In other words, theimport SimpleSchemaline in one package might be pulling in theSimpleSchemaobject from one package while that line in another package pulls in a completely differentSimpleSchemaobject. It will be difficult to know that this is happening unless you notice that your defaults are not being used by some of your schemas. To solve this, you can callSimpleSchema.constructorOptionDefaultsmultiple times or adjust your package dependencies to ensure that only one version ofsimpl-schemais pulled into your project.
import SimpleSchema from "simpl-schema";
const mySchema = new SimpleSchema({ name: String });
const doc = { name: 123 };
const cleanDoc = mySchema.clean(doc);
// cleanDoc is now mutated to hopefully have a better chance of passing validation
console.log(typeof cleanDoc.name); // stringWorks for a MongoDB update document (also known as "modifier" object), too:
import SimpleSchema from "simpl-schema";
const mySchema = new SimpleSchema({ name: String });
const updateDoc = { $set: { name: 123 } };
const cleanUpdateDoc = mySchema.clean(updateDoc);
// doc is now mutated to hopefully have a better chance of passing validation
console.log(typeof cleanUpdateDoc.$set.name); // stringLet's get into some more details about the different syntaxes that are supported when defining a schema. It's probably best to start with the simplest syntax. Here's an example:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  name: String,
  age: SimpleSchema.Integer,
  registered: Boolean,
});This is referred to as "shorthand" syntax. You simply map a property name to a type. When validating, SimpleSchema will make sure that all of those properties are present and are set to a value of that type.
In many cases, you will need to use longhand in order to define additional rules beyond what the data type should be.
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  name: {
    type: String,
    max: 40,
  },
  age: {
    type: SimpleSchema.Integer,
    optional: true,
  },
  registered: {
    type: Boolean,
    defaultValue: false,
  },
});You can use any combination of shorthand and longhand:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  name: String,
  age: {
    type: SimpleSchema.Integer,
    optional: true,
  },
  registered: Boolean,
});If you set the schema key to a regular expression, then the type will be String and the string must match the provided regular expression.
For example, this:
{
  exp: /foo/;
}is equivalent to:
{
  exp: { type: String, regEx: /foo/ }
}You can also set the schema key to an array of some type:
{
  friends: [String],
}is equivalent to:
{
  friends: { type: Array },
  'friends.$': { type: String },
}Note: This only applies to shorthand definitions, not to the longhand definition. This example will throw an error: { friends: { type: [String] } }.
You can define two or more different ways in which a key will be considered valid:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  id: SimpleSchema.oneOf(String, SimpleSchema.Integer),
  name: String,
});And this can be done in any mixture of shorthand and longhand:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  id: SimpleSchema.oneOf(
    {
      type: String,
      min: 16,
      max: 16,
    },
    {
      type: SimpleSchema.Integer,
      min: 0,
    }
  ),
  name: String,
});When one of the allowed types is an object, use a subschema. Don't mix the object property definitions in with the main schema.
Correct:
import SimpleSchema from "simpl-schema";
const objSchema = new SimpleSchema({
  _id: String,
});
const schema = new SimpleSchema({
  foo: SimpleSchema.oneOf(String, objSchema),
});Incorrect:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  foo: SimpleSchema.oneOf(String, Object),
  "foo._id": {
    type: String,
    optional: true,
  },
});NOTE: Multiple definitions is still an experimental feature and may not work as you expect in complex situations, such as where one of the valid definitions is an object or array. By reporting any weirdness you experience, you can help make it more robust.
If there are certain fields that are repeated in many of your schemas, it can be useful to define a SimpleSchema instance just for those fields and then merge them into other schemas:
import SimpleSchema from "simpl-schema";
import { idSchema, addressSchema } from "./sharedSchemas";
const schema = new SimpleSchema({
  name: String,
});
schema.extend(idSchema);
schema.extend(addressSchema);If the key appears in both schemas, the definition will be extended such that the result is the combination of both definitions.
import SimpleSchema from "simpl-schema";
import { idSchema, addressSchema } from "./sharedSchemas";
const schema = new SimpleSchema({
  name: {
    type: String,
    min: 5,
  },
});
schema.extend({
  name: {
    type: String,
    max: 15,
  },
});The above will result in the definition of the name field becoming:
{
  name: {
    type: String,
    min: 5,
    max: 15,
  },
}Note also that a plain object was passed to extend. If you pass a plain object, it is converted to a SimpleSchema instance for you.
Similar to extending, you can also reference other schemas as a way to define objects that occur within the main object:
import SimpleSchema from "simpl-schema";
import { addressSchema } from "./sharedSchemas";
const schema = new SimpleSchema({
  name: String,
  homeAddress: addressSchema,
  billingAddress: {
    type: addressSchema,
    optional: true,
  },
});Sometimes you have one large SimpleSchema object, and you need just a subset of it for some purpose.
To pull out certain schema keys into a new schema, you can use the pick method:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  firstName: String,
  lastName: String,
  username: String,
});
const nameSchema = schema.pick("firstName", "lastName");To keep all but certain keys in a new schema, you can use the omit method:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  firstName: String,
  lastName: String,
  username: String,
});
const nameSchema = schema.omit("username");To pull a subschema out of an Object key in a larger schema, you can use getObjectSchema:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  firstName: String,
  lastName: String,
  address: Object,
  "address.street1": String,
  "address.street2": { type: String, optional: true },
  "address.city": String,
  "address.state": String,
  "address.postalCode": String,
});
const addressSchema = schema.getObjectSchema("address");
// addressSchema is now the same as this:
// new SimpleSchema({
//   street1: String,
//   street2: { type: String, optional: true },
//   city: String,
//   state: String,
//   postalCode: String,
// });Sometimes if you want to get the rawDefinition of some schema just pass in the options { keepRawDefinition: true}(if not arg is passed the value will be null). Example:
const userSchema = new SimpleSchema(
  {
    name: String,
    number: "SimpleSchema.Integer",
    email: String,
  },
  { keepRawDefinition: true }
);
userSchema.rawDefinition;
//{
//   name: String,
//   number: 'SimpleSchema.Integer',
//   email: String
//}A basic schema key is just the name of the key (property) to expect in the objects that will be validated.
Use string keys with MongoDB-style dot notation to validate nested arrays and objects. For example:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  mailingAddress: Object,
  "mailingAddress.street": String,
  "mailingAddress.city": String,
});To indicate array items, use a $:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  addresses: {
    type: Array,
    minCount: 1,
    maxCount: 4,
  },
  "addresses.$": Object,
  "addresses.$.street": String,
  "addresses.$.city": String,
});Here are some specifics about the various rules you can define in your schema.
One of the following:
- String
- Number
- SimpleSchema.Integer(same as- Numberbut with decimals/floats disallowed)
- Boolean
- Object
- Array
- Any custom or built-in class like Date
- Another SimpleSchemainstance, meaningObjecttype with this schema
- SimpleSchema.oneOf(...), with multiple of the above types
- SimpleSchema.Any
Can also be a function that returns the label
A string that will be used to refer to this field in validation error messages. The default is an inflected (humanized) derivation of the key name itself. For example, the key "firstName" will have a default label of "First name" if you do not include the label property in your definition.
You can use the labels function to alter one or more labels on the fly:
schema.labels({
  password: "Enter your password",
});To get the label for a field, use schema.label(fieldName), which returns a usable string.
Can also be a function that returns true or false
By default, all keys are required. Set optional: true to change that.
With complex keys, it might be difficult to understand what "required" means. Here's a brief explanation of how requiredness is interpreted:
- If typeisArray, then "required" means that key must have a value, but an empty array is fine. (If an empty array is not fine, add theminCount: 1option.)
- For array items (when the key name ends with ".$"), if optionalis true, thennullvalues are valid. If array items are required, then anynullitems will fail the type check.
- If a key is required at a deeper level, the key must have a value only if the object it belongs to is present.
- When the object being validated is a MongoDB update document (also known as "modifier" object), changes that would unset or nulla required key result in validation errors.
That last point can be confusing, so let's look at a couple examples:
- Say you have a required key "friends.address.city" but "friends.address" is optional. If "friends.address" is set in the object you're validating, but "friends.address.city" is not, there is a validation error. However, if "friends.address" is not set, then there is no validation error for "friends.address.city" because the object it belongs to is not present.
- If you have a required key "friends.$.name", but the friendsarray has no objects in the object you are validating, there is no validation error for "friends.$.name". When thefriendsarray does have objects, every present object is validated, and each object could potentially have a validation error if it is missing thenameproperty. For example, when there are two objects in the friends array and both are missing thenameproperty, there will be a validation error for both "friends.0.name" and "friends.1.name".
Can also be a function that returns true or false
If you would rather have all your schema keys be optional by default, pass the requiredByDefault: false option and then use required: true to make individual keys required.
const schema = new SimpleSchema(
  {
    optionalProp: String,
    requiredProp: { type: String, required: true },
  },
  { requiredByDefault: false }
);Can also be a function that returns the min/max value
- If typeisNumberorSimpleSchema.Integer, these rules define the minimum or maximum numeric value.
- If typeisString, these rules define the minimum or maximum string length.
- If typeisDate, these rules define the minimum or maximum date, inclusive.
You can alternatively provide a function that takes no arguments and returns the appropriate minimum or maximum value. This is useful, for example, if the minimum Date for a field should be "today".
Can also be a function that returns true or false
Set to true to indicate that the range of numeric values, as set by min/max, are to be treated as an exclusive range. Set to false (default) to treat ranges as inclusive.
Can also be a function that returns the minCount/maxCount value
Define the minimum or maximum array length. Used only when type is Array.
Can also be a function that returns the array or the Set of allowed values
An array or a Set of values that are allowed. A key will be invalid if its value is not one of these.
You can use schema.getAllowedValuesForKey(key) to get the allowed values array for a key.
Note: If you wish to restrict the items allowed in an Array, the allowedValues property must be on the array item definition.
const schema = new SimpleSchema({
  myArray: {
    type: Array,
  },
  "myArray.$": {
    type: String,
    allowedValues: ["foo", "bar"],
  },
});Can also be a function that returns a regular expression or an array of them
Any regular expression that must be matched for the key to be valid, or an array of regular expressions that will be tested in order.
In earlier releases, the SimpleSchema.RegEx object defined standard regular expressions that you could use as the value for the regEx key. However, many of these were prone to DDoS security risks, causing this package to be flagged by security audit tooling. We no longer include any built-in regular expressions. We encourage you to use a custom function or some other method of validating any strings with complex requirements. If necessary, you can look back at the code for previous SimpleSchema releases and copy specific built-in regular expressions into your codebase.
Also, check out recheck, which includes an ESLint plugin you can add to your project to avoid accidentally writing regular expressions that could be attacked.
Can also be a function that returns true or false
Set to true when regEx is set if you want an empty string to always pass validation, even though the regular expression may disallow it.
If you have a key with type Object, the properties of the object will be validated as well, so you must define all allowed properties in the schema. If this is not possible or you don't care to validate the object's properties, use the blackbox: true option to skip validation for everything within the object.
Prior to SimpleSchema 2.0, objects that are instances of a custom class were considered to be blackbox by default. This is no longer true, so if you do not want your class instance validated, be sure to add blackbox: true in your schema.
Used by the cleaning process but not by validation
When you call simpleSchemaInstance.clean() with trimStrings set to true, all string values are trimmed of leading and trailing whitespace. If you set trim to false for certain keys in their schema definition, those keys will be skipped.
Refer to the Custom Validation section.
Used by the cleaning process but not by validation
Set this to any value that you want to be used as the default when an object does not include this field or has this field set to undefined. This value will be injected into the object by a call to mySimpleSchema.clean() with getAutovalues: true.
Note the following points of confusion:
- A default value itself is not cleaned. So, for example, if your default value is "", it will not be removed by the removeEmptyStringsoperation in the cleaning.
- A default value is added only if there isn't a value set AND the parent object exists. Usually this is what you want, but if you need to ensure that it will always be added, you can add defaultValue: {}to all ancestor objects.
If you need more control, use the autoValue option instead.
To get the defaultValue for a field, use schema.defaultValue(fieldName). It is a shorthand for schema.get(fieldName, 'defaultValue').
Used by the cleaning process but not by validation
The autoValue option allows you to specify a function that is called by simpleSchemaInstance.clean() to potentially change the value of a property in the object being cleaned. This is a powerful feature that allows you to set up either forced values or default values, potentially based on the values of other fields in the object.
An autoValue function this context provides a variety of properties and methods to help you determine what you should return:
- this.closestSubschemaFieldName: If your schema is used as a subschema in another schema, this will be set to the name of the key that references the schema. Otherwise it will be- null.
- this.field(): Use this method to get information about other fields. Pass a field name (schema key) as the only argument. The return object will have- isSet,- value, and- operatorproperties for that field.
- this.genericKey: The generic schema key for which the autoValue is running (- $in place of actual array index).
- this.isInArrayItemObject: True if we're traversing an object that's in an array.
- this.isInSubObject: True if we're traversing an object that's somewhere within another object.
- this.isModifier: True if this is running on a MongoDB update document (also known as "modifier" object).
- this.isSet: True if the field is already set in the document
- this.key: The schema key for which the autoValue is running. This is usually known, but if your autoValue function is shared among various keys or if your schema is used as a subschema in another schema, this can be useful.
- this.obj: The full object.
- this.operator: If isSet = true and isUpdate = true, this contains the name of the update operator in the update document in which this field is being changed. For example, if the update document were- {$set: {name: "Alice"}}, in the autoValue function for the- namefield,- this.isSetwould be true,- this.valuewould be "Alice", and- this.operatorwould be "$set".
- this.parentField(): Use this method to get information about the parent object. Works the same way as- field().
- this.siblingField(): Use this method to get information about other fields that have the same parent object. Works the same way as- field(). This is helpful when you use sub-schemas or when you're dealing with arrays of objects.
- this.unset(): Call this method to prevent the original value from being used when you return undefined.
- this.value: If isSet = true, this contains the field's current (requested) value in the document.
If an autoValue function does not return anything (i.e., returns undefined), the field's value will be whatever the document says it should be. If that field is already in the document, it stays in the document with the same value. If it's not in the document, it's still not there. If you don't want it to be in the doc, you must call this.unset().
Any other return value will be used as the field's value. You may also return special pseudo-modifier objects for update operations. Examples are {$inc: 1} and {$push: new Date}.
- If your autoValue for one field relies on the autoValue or defaultValue of another field, make sure that the other field is listed before the field that relies on it in the schema. autoValues are run in order from least nested, to most nested, so you can assume that parent values will be set, but for fields at the same level, schema order matters. Refer to issue #204.
- An autoValuefunction will always run during cleaning even if that field is not in the object being cleaned. This allows you to provide complex default values. If your function applies only when there is a value, you should addif (!this.isSet) return;at the top.
You may have noticed that many of the rule properties can be set to functions that return the value. If you do this, the this context within those functions will have the following properties:
- this.field(): Use this method to get information about other fields. Pass a field name (schema key) as the only argument. The return object will have- isSet,- value, and- operatorproperties for that field.
- this.genericKey: The generic schema key for which the autoValue is running (- $in place of actual array index).
- this.isInArrayItemObject: True if we're traversing an object that's in an array.
- this.isInSubObject: True if we're traversing an object that's somewhere within another object.
- this.isModifier: True if this is running on a MongoDB update document (also known as "modifier" object).
- this.isSet: True if the field is already set in the document
- this.key: The schema key for which the autoValue is running. This is usually known, but if your autoValue function is shared among various keys or if your schema is used as a subschema in another schema, this can be useful.
- this.obj: The full object.
- this.operator: If isSet = true and isUpdate = true, this contains the name of the update operator in the update document in which this field is being changed. For example, if the update document were- {$set: {name: "Alice"}}, in the autoValue function for the- namefield,- this.isSetwould be true,- this.valuewould be "Alice", and- this.operatorwould be "$set".
- this.parentField(): Use this method to get information about the parent object. Works the same way as- field().
- this.siblingField(): Use this method to get information about other fields that have the same parent object. Works the same way as- field(). This is helpful when you use sub-schemas or when you're dealing with arrays of objects.
- this.validationContext: The current validation context
- this.value: If isSet = true, this contains the field's current (requested) value in the document.
To obtain field's property value, just call the get method.
const schema = new SimpleSchema({
  friends: {
    type: Array,
    minCount: 0,
    maxCount: 3,
  },
});
schema.get("friends", "maxCount"); // 3The object you pass in when validating can be a normal object, an instance of any class, or a MongoDB update document (also known as "modifier" object). It can also be an array of any of these. In other words, you can pass
in the exact reference that you are going to pass to insertOne, insertMany, updateOne, or updateMany. (This is what the Collection2 Meteor package does for you.)
There are three ways to validate an object against your schema:
- With a throwaway context, throwing an Error for the first validation error found (schema.validate())
- With a unique unnamed validation context, not throwing any Errors (schema.newContext().validate())
- With a unique named validation context, not throwing any Errors (schema.namedContext('someUniqueString').validate())
- With the default validation context, not throwing any Errors. (schema.namedContext().validate())
A validation context provides methods for validating and checking the validation status of a particular object.
It's usually best to use a named validation context. That way, the context is automatically persisted by name, allowing you to easily rely on its methods.
Here is an example of obtaining a named validation context:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  name: String,
});
const userFormValidationContext = schema.namedContext("userForm");The first time you request a context with a certain name, it is created. Calling namedContext() passing no arguments is equivalent to calling namedContext('default').
To obtain an unnamed validation context, call newContext():
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  name: String,
});
const myValidationContext = schema.newContext();An unnamed validation context is not persisted anywhere. It can be useful when you need to see if a document is valid but you don't need any of the methods for that context, or if you are going to keep the context reference in memory yourself.
To validate an object against the schema in a validation context, call validationContextInstance.validate(obj, options). This method returns true if the object is valid according to the schema or false if it is not. It also stores a list of invalid fields and corresponding error messages in the context object.
You can call myContext.isValid() to see if the object last passed into validate() was found to be valid. Returns true or false.
For a list of options, see the Validation Options section.
You may have the need to (re)validate certain keys while leaving any errors for other keys unchanged. For example, if you have several errors on a form and you want to revalidate only the invalid field the user is currently typing in. For this situation, call myContext.validate with the keys option set to an array of keys that should be validated.
This method returns true only if all the specified schema keys and their descendent keys are valid according to the schema. Otherwise it returns false.
As a convenience, you may pass an array of objects to the validate function and SimpleSchema will validate them all. The first error found will cause the whole array to be considered invalid. The array itself is not validated, so validation will pass if it is empty and field names will not begin with $.
validate() accepts the following options:
- modifier: Are you validating a Mongo modifier object? False by default.
- upsert: Are you validating a Mongo modifier object potentially containing upsert operators? False by default.
- extendedCustomContext: This object will be added to the- thiscontext in any custom validation functions that are run during validation. See the Custom Validation section.
- ignore: An array of validation error types (in SimpleSchema.ErrorTypes enum) to ignore.
- keys: An array of keys to validate. If not provided, revalidates the entire object.
- Call mySimpleSchema.validate(obj, options)to validateobjagainst the schema and throw aValidationErrorif invalid.
- Call SimpleSchema.validate(obj, schema, options)static function as a shortcut formySimpleSchema.validateif you don't want to createmySimpleSchemafirst. Theschemaargument can be just the schema object, in which case it will be passed to theSimpleSchemaconstructor for you. This is likecheck(obj, schema)but without thecheckdependency and with the ability to pass full schema error details back to a callback on the client.
- Call mySimpleSchema.validator()to get a function that callsmySimpleSchema.validatefor whatever object is passed to it. This means you can dovalidate: mySimpleSchema.validator()in the mdg:validated-method package.
- Call mySimpleSchema.getFormValidator()to get a function that validates whatever object is passed to it and returns a Promise that resolves with errors. The returned function is compatible with the Composable Form Specification.
You can defineValidationErrorTransform one time somewhere in your code to customize the error or change it to a more specific type.
import SimpleSchema from "simpl-schema";
SimpleSchema.defineValidationErrorTransform((error) => {
  const customError = new MyCustomErrorType(error.message);
  customError.errorList = error.details;
  return customError;
});For example, in a Meteor app, in order to ensure that the error details are sent back to the client when throwing an error in a server method, you can convert it to a Meteor.Error:
import SimpleSchema from "simpl-schema";
SimpleSchema.defineValidationErrorTransform((error) => {
  const ddpError = new Meteor.Error(error.message);
  ddpError.error = "validation-error";
  ddpError.details = error.details;
  return ddpError;
});There are three ways to attach custom validation methods.
To add a custom validation function that is called for ALL keys in ALL schemas (for example, to publish a package that adds global support for some additional rule):
SimpleSchema.addValidator(myFunction);To add a custom validation function that is called for ALL keys for ONE schema:
import SimpleSchema from 'simpl-schema';
const schema = new SimpleSchema({ ... });
schema.addValidator(myFunction);To add a custom validation function that is called for ONE key in ONE schema:
import SimpleSchema from "simpl-schema";
const schema = new SimpleSchema({
  someKey: {
    type: String,
    custom: myFunction,
  },
});All custom validation functions work the same way. First, do the necessary custom validation, use this to get whatever information you need. Then, if valid, return undefined. If invalid, return an error type string. The error type string can be one of the built-in strings or any string you want.
- If you return a built-in string, it's best to use the SimpleSchema.ErrorTypesconstants.
- If you return a custom string, you'll usually want to define a message for it.
Within your custom validation function, this provides the following properties:
- key: The name of the schema key (e.g., "addresses.0.street")
- genericKey: The generic name of the schema key (e.g., "addresses.$.street")
- definition: The schema definition object.
- isSet: Does the object being validated have this key set?
- value: The value to validate.
- operator: The Mongo operator for which we're doing validation. Might be- null.
- validationContext: The current- ValidationContextinstance
- field(): Use this method to get information about other fields. Pass a field name (non-generic schema key) as the only argument. The return object will have- isSet,- value, and- operatorproperties for that field.
- siblingField(): Use this method to get information about other fields that have the same parent object. Works the same way as- field(). This is helpful when you use sub-schemas or when you're dealing with arrays of objects.
- addValidationErrors(errors): Call this to add validation errors for any key. In general, you should use this to add errors for other keys. To add an error for the current key, return the error type string. If you do use this to add an error for the current key, return- falsefrom your custom validation function.
NOTE: If you need to do some custom validation on the server and then display errors back on the client, refer to the Asynchronous Custom Validation on the Client section.
Add a validator for all schemas:
import SimpleSchema from "simpl-schema";
SimpleSchema.addDocValidator((obj) => {
  // Must return an array, potentially empty, of objects with `name` and `type` string properties and optional `value` property.
  return [{ name: "firstName", type: "TOO_SILLY", value: "Reepicheep" }];
});Add a validator for one schema:
import SimpleSchema from 'simpl-schema';
const schema = new SimpleSchema({ ... });
schema.addDocValidator(obj => {
  // Must return an array, potentially empty, of objects with `name` and `type` string properties and optional `value` property.
  return [
    { name: 'firstName', type: 'TOO_SILLY', value: 'Reepicheep' }
  ];
});Whole-document validators have the following available on this context:
- this.ignoreTypes: The value of the- ignoreoption that was passed to- validate.
- this.isModifier: True if this is running on a MongoDB modifier object.
- this.isUpsert: True if this is running on a MongoDB modifier object that is for an upsert.
- this.keysToValidate: The value of the- keysoption that was passed to- validate.
- this.mongoObject: The- MongoObjectinstance.
- this.obj: The full object.
- this.schema: The schema instance.
- this.validationContext: The- ValidationContextinstance.
If you want to display an arbitrary validation error and it is not possible to use a custom validation function (perhaps you have to call a function onSubmit or wait for asynchronous results), you can add one or more errors to a validation context at any time by calling myContext.addValidationErrors(errors), where errors is an array of error objects with the following format:
{name: key, type: errorType, value: anyValue}- name: The schema key as specified in the schema.
- type: The type of error. Any string you want, or one of the strings in the- SimpleSchema.ErrorTypeslist.
- value: Optional. The value that was not valid. Will be used to replace the- [value]placeholder in error messages.
If you use a custom string for type, be sure to register a getErrorMessage function. (See Customizing Validation Messages).
Example:
myValidationContext.addValidationErrors([
  { name: "password", type: "wrongPassword" },
]);Call myValidationContext.validationErrors() to get the full array of validation errors. Each object in the array has at least two keys:
- name: The schema key as specified in the schema.
- type: The type of error. See- SimpleSchema.ErrorTypes.
There may also be a value property, which is the value that was invalid.
There may be a message property, but usually the error message is constructed from message templates. You should call ctxt.keyErrorMessage(key) to get a message string rather than using error.message directly.
The built-in validation errors have built-in English messages associated with them. However, if you have custom validation error types, need messages in other languages, or just want to change some default messages, you will want to register a getErrorMessage function for your schema.
const schema = new SimpleSchema({
  name: String
}, {
  getErrorMessage(error, label) {
    if (error.type === 'too_long') return `${label} is too long!`
    // Returning undefined will fall back to using defaults
  }
})You can also do this globally for all schemas:
globalThis.simpleSchemaGlobalConfig = {
  getErrorMessage(error, label) {
    if (error.type === 'too_long') return `${label} is too long!`
    // Returning undefined will fall back to using defaults
  }
};A getErrorMessage function in schema options will be tried before a global getErrorMessage function. If the schema getErrorMessage returns undefined, the global getErrorMessage will be called, and if that returns undefined, the built-in English message will be used.
myContext.keyIsInvalid(key) returns true if the specified key is currently
invalid, or false if it is valid.
myContext.keyErrorMessage(key) returns the error message for the specified
key if it is invalid. If it is valid, this method returns an empty string.
Call myContext.reset() if you need to reset the validation context, clearing out any invalid field messages and making it valid.
myContext.name is set to the context name, if it is a named context. Create named contexts by calling schema.namedContext(name) or new ValidationContext(schema, name).
Call MySchema.schema([key]) to get the schema definition object. If you specify a key, then only the schema definition for that key is returned.
Note that this may not match exactly what you passed into the SimpleSchema constructor. The schema definition object is normalized internally, and this method returns the normalized copy.
You can call simpleSchemaInstance.clean() or simpleSchemaValidationContextInstance.clean() to clean the object you're validating. Do this prior to validating it to avoid any avoidable validation errors.
The clean function takes the object to be cleaned as its first argument and the following optional options as its second argument:
- mutate: The object is copied before being cleaned. If you don't mind mutating the object you are cleaning, you can pass- mutate: trueto get better performance.
- isModifier: Is the first argument a modifier object? False by default.
- filter:- trueby default. If- true, removes any keys not explicitly or implicitly allowed by the schema, which prevents errors being thrown for those keys during validation.
- autoConvert:- trueby default. If- true, helps eliminate unnecessary validation messages by automatically converting values where possible.- Non-string values are converted to a String if the schema expects a String
- Strings that are numbers are converted to Numbers if the schema expects a Number
- Strings that are "true" or "false" are converted to Boolean if the schema expects a Boolean
- Numbers are converted to Boolean if the schema expects a Boolean, with 0 being falseand all other numbers beingtrue
- Non-array values are converted to a one-item array if the schema expects an Array
 
- removeEmptyStrings: Remove keys in normal object or $set where the value is an empty string? True by default.
- trimStrings: Remove all leading and trailing spaces from string values? True by default.
- getAutoValues: Run- autoValuefunctions and inject automatic and- defaultValuevalues? True by default.
- extendAutoValueContext: This object will be added to the- thiscontext of autoValue functions.- extendAutoValueContextcan be used to give your- autoValuefunctions additional valuable information, such as- userId. (Note that operations done using the Collection2 package automatically add- userIdto the- autoValuecontext already.)
You can also set defaults for any of these options in your SimpleSchema constructor options:
const schema = new SimpleSchema(
  {
    name: String,
  },
  {
    clean: {
      trimStrings: false,
    },
  }
);NOTE: The Collection2 package always calls clean before every insert, update, or upsert.
For consistency, if you care only about the date (year, month, date) portion and not the time, then use a Date object set to the desired date at midnight UTC (note, the clean function won't strip out time). This goes for min and max dates, too. If you care only about the date
portion and you want to specify a minimum date, min should be set to midnight UTC on the minimum date (inclusive).
Following these rules ensures maximum interoperability with HTML5 date inputs and usually just makes sense.
If you have a field that should be required only in certain circumstances, first make the field optional, and then use a custom function similar to this:
{
  field: {
    type: String,
    optional: true,
    custom: function () {
      let shouldBeRequired = this.field('saleType').value === 1;
      if (shouldBeRequired) {
        // inserts
        if (!this.operator) {
          if (!this.isSet || this.value === null || this.value === "") return SimpleSchema.ErrorTypes.REQUIRED;
        }
        // updates
        else if (this.isSet) {
          if (this.operator === "$set" && this.value === null || this.value === "") return SimpleSchema.ErrorTypes.REQUIRED;
          if (this.operator === "$unset") return SimpleSchema.ErrorTypes.REQUIRED;
          if (this.operator === "$rename") return SimpleSchema.ErrorTypes.REQUIRED;
        }
      }
    }
  }
}Where customCondition is whatever should trigger it being required.
Here's an example of declaring one value valid or invalid based on another value using a custom validation function.
MySchema = new SimpleSchema({
  password: {
    type: String,
    label: "Enter a password",
    min: 8,
  },
  confirmPassword: {
    type: String,
    label: "Enter the password again",
    min: 8,
    custom() {
      if (this.value !== this.field("password").value) {
        return "passwordMismatch";
      }
    },
  },
});Set SimpleSchema.debug = true in your app before creating any named
validation contexts to cause all named validation contexts to automatically
log all invalid key errors to the browser console. This can be helpful while
developing an app to figure out why certain actions are failing validation.
You may find at some point that there is something extra you would really like to define within a schema for your package or app. However, if you add unrecognized options to your schema definition, you will get an error. To inform SimpleSchema about your custom option and avoid the error, you need to call SimpleSchema.extendOptions. By way of example, here is how the Collection2 package adds the additional schema options it provides:
SimpleSchema.extendOptions(["index", "unique", "denyInsert", "denyUpdate"]);Obviously you need to ensure that extendOptions is called before any SimpleSchema instances are created with those options.
import { toJsonSchema } from 'simpl-schema'
const schema = new SimpleSchema({
  name: String
})
const jsonSchema = toJsonSchema(schema)mxab:simple-schema-jsdoc Generate jsdoc from your schemas.
(Submit a PR to list your package here)
This project exists thanks to all the people who contribute. [Contribute].
You can support this project by becoming a sponsor.
MIT
Anyone is welcome to contribute. Before submitting a pull request, make sure that you've added tests for your changes, and that all tests pass when you run npm test.
(Add your name if it's missing.)
- @mquandalle
- @Nemo64
- @DavidSichau