- Joi
validate(value, schema, [options], [callback])
compile(schema)
assert(value, schema, [message])
attempt(value, schema, [message])
ref(key, [options])
isRef(ref)
reach(schema, path)
extend(extension)
any
any.allow(value)
any.valid(value)
- aliases:only
,equal
any.invalid(value)
- aliases:disallow
,not
any.required()
any.optional()
any.forbidden()
any.strip()
any.description(desc)
any.notes(notes)
any.tags(tags)
any.meta(meta)
any.example(value)
any.unit(name)
any.options(options)
any.strict(isStrict)
any.default([value, [description]])
any.concat(schema)
any.when(ref, options)
any.label(name)
any.raw(isRaw)
any.empty(schema)
any.error(err)
array
boolean
binary
date
func
number
object
object.keys([schema])
object.min(limit)
object.max(limit)
object.length(limit)
object.pattern(regex, schema)
object.and(peers)
object.nand(peers)
object.or(peers)
object.xor(peers)
object.with(key, peers)
object.without(key, peers)
object.rename(from, to, [options])
object.assert(ref, schema, [message])
object.unknown([allow])
object.type(constructor, [name])
object.schema()
object.requiredKeys(children)
object.optionalKeys(children)
string
string.insensitive()
string.min(limit, [encoding])
string.max(limit, [encoding])
string.truncate([enabled])
string.creditCard()
string.length(limit, [encoding])
string.regex(pattern, [name])
string.replace(pattern, replacement)
string.alphanum()
string.token()
string.email([options])
string.ip([options])
string.uri([options])
string.guid()
string.hex()
string.hostname()
string.lowercase()
string.uppercase()
string.trim()
string.isoDate()
alternatives
lazy(fn)
- Errors
Validates a value using the given schema and options where:
value
- the value being validated.schema
- the validation schema. Can be a joi type object or a plain object where every key is assigned a joi type object.options
- an optional object with the following optional keys:abortEarly
- whentrue
, stops validation on the first error, otherwise returns all the errors found. Defaults totrue
.convert
- whentrue
, attempts to cast values to the required types (e.g. a string to a number). Defaults totrue
.allowUnknown
- whentrue
, allows object to contain unknown keys which are ignored. Defaults tofalse
.skipFunctions
- whentrue
, ignores unknown keys with a function value. Defaults tofalse
.stripUnknown
- remove unknown elements from objects and arrays. Defaults tofalse
.- when
true
, all unknown elements will be removed. - when an
object
:arrays
- set totrue
to remove unknown items from arrays.objects
- set totrue
to remove unknown keys from objects.
- when
language
- overrides individual error messages. Defaults to no override ({}
). Messages apply the following rules :- variables are put between curly braces like
{{var}}
, if prefixed by a!
like{{!var}}
, it will be html escaped - strings are always preceeded by the key name, unless a
{{key}}
is found elsewhere or if the string is prefixed by a!!
- when
'label'
is set, it overrides the key name in the error message - to better understand the structure of the language, it's advised to have a look at the existing messages you want to override here
- variables are put between curly braces like
presence
- sets the default presence requirements. Supported modes:'optional'
,'required'
, and'forbidden'
. Defaults to'optional'
.context
- provides an external data set to be used in references. Can only be set as an external option tovalidate()
and not usingany.options()
.noDefaults
- whentrue
, do not apply default values. Defaults tofalse
.error
- overrides the default joi error as described inany.error(err)
. Defaults to no override.
callback
- the optional synchronous callback method using the signaturefunction(err, value)
where:err
- if validation failed, the error reason, otherwisenull
.value
- the validated value with any type conversions and other modifiers applied (the input is left unchanged).value
can be incomplete if validation failed andabortEarly
istrue
. If callback is not provided, then returns an object with error and value properties.
const schema = {
a: Joi.number()
};
const value = {
a: '123'
};
Joi.validate(value, schema, (err, value) => { });
// err -> null
// value.a -> 123 (number, not string)
// or
const result = Joi.validate(value, schema);
// result.error -> null
// result.value -> { "a" : 123 }
Converts literal schema definition to joi schema object (or returns the same back if already a joi schema object) where:
schema
- the schema definition to compile.
const definition = ['key', 5, { a: true, b: [/^a/, 'boom'] }];
const schema = Joi.compile(definition);
// Same as:
const schema = Joi.alternatives().try([
Joi.string().valid('key'),
Joi.number().valid(5),
Joi.object().keys({
a: Joi.boolean().valid(true),
b: Joi.alternatives().try([
Joi.string().regex(/^a/),
Joi.string().valid('boom')
])
})
]);
Validates a value against a schema and throws if validation fails where:
value
- the value to validate.schema
- the schema object.message
- optional message string prefix added in front of the error message. may also be an Error object.
Joi.assert('x', Joi.number());
Validates a value against a schema, returns valid object, and throws if validation fails where:
value
- the value to validate.schema
- the schema object.message
- optional message string prefix added in front of the error message. may also be an Error object.
Joi.attempt('x', Joi.number()); // throws error
const result = Joi.attempt('4', Joi.number()); // result -> 4
Generates a reference to the value of the named key. References are resolved at validation time and in order of dependency so that if one key validation depends on another, the dependent key is validated second after the reference is validated. References support the following arguments:
key
- the reference target. References cannot point up the object tree, only to sibling keys, but they can point to their siblings' children (e.g. 'a.b.c') using the.
separator. If akey
starts with$
is signifies a context reference which is looked up in thecontext
option object.options
- optional settings:separator
- overrides the default.
hierarchy separator.contextPrefix
- overrides the default$
context prefix signifier.- Other options can also be passed based on what
Hoek.reach
supports.
Note that references can only be used where explicitly supported such as in valid()
or invalid()
rules. If upwards
(parents) references are needed, use object.assert()
.
const schema = Joi.object().keys({
a: Joi.ref('b.c'),
b: {
c: Joi.any()
},
c: Joi.ref('$x')
});
Joi.validate({ a: 5, b: { c: 5 } }, schema, { context: { x: 5 } }, (err, value) => {});
Checks whether or not the provided argument is a reference. It's especially useful if you want to post-process error messages.
const ref = Joi.ref('a');
Joi.isRef(ref); // returns true
Get a sub-schema of an existing schema based on a path. Path separator is a dot (.
).
const schema = Joi.object({ foo: Joi.object({ bar: Joi.number() }});
const number = Joi.reach(schema, 'foo.bar');
Creates a new Joi instance customized with the extension(s) you provide included.
It is important to understand that original Joi library is not modified by this.
The extension makes use of some common structures that need to be described prior :
value
- the value being processed by Joi.state
- an object containing the current context of validation.key
- the key of the current value.path
- the full path of the current value.parent
- the potential parent of the current value.
options
- options object provided throughany().options()
orJoi.validate()
.
extension
can be a single extension object or an array of extension objects using the following parameters :
name
- name of the new type you are defining, this can be an existing type. Required.base
- an existing Joi schema to base your type upon. Defaults toJoi.any()
.coerce
- an optional function that runs before the base, usually serves when you want to coerce values of a different type than your base. It takes 3 argumentsvalue
,state
andoptions
.pre
- an optional function that runs first in the validation chain, usually serves when you need to cast values. It takes 3 argumentsvalue
,state
andoptions
.language
- an optional object to add error definitions. Every key will be prefixed by the type name.describe
- an optional function taking the fully formed description to post-process it.rules
- an optional array of rules to add.name
- name of the new rule. Required.params
- an optional object containing Joi schemas of each parameter ordered. You can also pass a single Joi schema as long as it is aJoi.object()
, of course some methods such aspattern
orrename
won't be useful or won't work at all in this given context.setup
- an optional function that takes an object with the provided parameters. One ofsetup
orvalidate
must be provided.validate
- an optional function to validate values that takes 4 parametersparams
,value
,state
andoptions
. One ofsetup
orvalidate
must be provided.description
- an optional string or function taking the parameters as argument to describe what the rule is doing.
If you publish your extension on npm, make sure to add joi
and extension
as keywords so that it's discoverable more easily.
const Joi = require('joi');
const customJoi = Joi.extend({
base: Joi.number(),
name: 'number',
language: {
round: 'needs to be a rounded number', // Used below as 'number.round'
dividable: 'needs to be dividable by {{q}}'
},
pre(value, state, options) {
if (options.convert && this._flags.round) {
return Math.round(value); // Change the value
}
return value; // Keep the value as it was
},
rules: [
{
name: 'round',
setup(params) {
this._flags.round = true; // Set a flag for later use
},
validate(params, value, state, options) {
if (value % 1 !== 0) {
// Generate an error, state and options need to be passed
return this.createError('number.round', { v: value }, state, options);
}
return value; // Everything is OK
}
},
{
name: 'dividable',
params: {
q: Joi.alternatives([Joi.number().required(), Joi.func().ref()])
},
validate(params, value, state, options) {
if (value % params.q !== 0) {
// Generate an error, state and options need to be passed, q is used in the language
return this.createError('number.dividable', { v: value, q: params.q }, state, options);
}
return value; // Everything is OK
}
}
]
});
const schema = customJoi.number().round().dividable(3);
Generates a schema object that matches any data type.
const any = Joi.any();
any.validate('a', (err, value) => { });
Whitelists a value where:
value
- the allowed value which can be of any type and will be matched against the validated value before applying any other rules.value
can be an array of values, or multiple values can be passed as individual arguments.value
supports references.
Note that this whitelist of allowed values is in addition to any other permitted values.
To create an exclusive whitelist of values, see any.valid(value)
.
const schema = {
a: Joi.any().allow('a'),
b: Joi.any().allow('b', 'B'),
c: Joi.any().allow(['c', 'C'])
};
Adds the provided values into the allowed whitelist and marks them as the only valid values allowed where:
value
- the allowed value which can be of any type and will be matched against the validated value before applying any other rules.value
can be an array of values, or multiple values can be passed as individual arguments.value
supports references.
const schema = {
a: Joi.any().valid('a'),
b: Joi.any().valid('b', 'B'),
c: Joi.any().valid(['c', 'C'])
};
Blacklists a value where:
value
- the forbidden value which can be of any type and will be matched against the validated value before applying any other rules.value
can be an array of values, or multiple values can be passed as individual arguments.value
supports references.
const schema = {
a: Joi.any().invalid('a'),
b: Joi.any().invalid('b', 'B'),
c: Joi.any().invalid(['c', 'C'])
};
Marks a key as required which will not allow undefined
as value. All keys are optional by default.
const schema = Joi.any().required();
Marks a key as optional which will allow undefined
as values. Used to annotate the schema for readability as all keys are optional by default.
const schema = Joi.any().optional();
Marks a key as forbidden which will not allow any value except undefined
. Used to explicitly forbid keys.
const schema = {
a: Joi.any().forbidden()
};
Marks a key to be removed from a resulting object or array after validation. Used to sanitize output.
const schema = Joi.object({
username: Joi.string(),
password: Joi.string().strip()
});
schema.validate({ username: 'test', password: 'hunter2' }, (err, value) => {
// value = { username: 'test' }
});
const schema = Joi.array().items(Joi.string(), Joi.any().strip());
schema.validate(['one', 'two', true, false, 1, 2], (err, value) => {
// value = ['one', 'two']
});
Annotates the key where:
desc
- the description string.
const schema = Joi.any().description('this key will match anything you give it');
Annotates the key where:
notes
- the notes string or array of strings.
const schema = Joi.any().notes(['this is special', 'this is important']);
Annotates the key where:
tags
- the tag string or array of strings.
const schema = Joi.any().tags(['api', 'user']);
Attaches metadata to the key where:
meta
- the meta object to attach.
const schema = Joi.any().meta({ index: true });
Annotates the key where:
value
- an example value.
If the example fails to pass validation, the function will throw.
const schema = Joi.string().min(4).example('abcd');
Annotates the key where:
name
- the unit name of the value.
const schema = Joi.number().unit('milliseconds');
Overrides the global validate()
options for the current key and any sub-key where:
options
- an object with the same optional keys asJoi.validate(value, schema, options, callback)
.
const schema = Joi.any().options({ convert: false });
Strict mode sets the options.convert
options to false
which prevent type casting for the current key and any child keys.
isStrict
- whether strict mode is enabled or not. Defaults to true.
const schema = Joi.any().strict();
Sets a default value if the original value is undefined where:
value
- the value.value
supports references.value
may also be a function which returns the default value. Ifvalue
is specified as a function that accepts a single parameter, that parameter will be a context object that can be used to derive the resulting value. This clones the object however, which incurs some overhead so if you don't need access to the context define your method so that it does not accept any parameters.- without any
value
,default
has no effect, except forobject
that will then create nested defaults (applying inner defaults of that object).
Note that if value
is an object, any changes to the object after default()
is called will change the reference
and any future assignment.
Additionally, when specifying a method you must either have a description
property on your method or the second parameter is required.
const generateUsername = (context) => {
return context.firstname.toLowerCase() + '-' + context.lastname.toLowerCase();
};
generateUsername.description = 'generated username';
const schema = {
username: Joi.string().default(generateUsername),
firstname: Joi.string(),
lastname: Joi.string(),
created: Joi.date().default(Date.now, 'time of creation'),
status: Joi.string().default('registered')
};
Joi.validate({
firstname: 'Jane',
lastname: 'Doe'
}, schema, (err, value) => {
// value.status === 'registered'
// value.username === 'jane-doe'
// value.created will be the time of validation
});
Returns a new type that is the result of adding the rules of one type to another where:
schema
- a joi type to merge into the current schema. Can only be of the same type as the context type orany
. If applied to anany
type, the schema can be any other schema.
const a = Joi.string().valid('a');
const b = Joi.string().valid('b');
const ab = a.concat(b);
Converts the type into an alternatives
type where the conditions are merged into the type definition where:
ref
- the key name or reference.options
- an object with:is
- the required condition joi type.then
- the alternative schema type if the condition is true. Required ifotherwise
is missing.otherwise
- the alternative schema type if the condition is false. Required ifthen
is missing.
const schema = {
a: Joi.any().valid('x').when('b', { is: 5, then: Joi.valid('y'), otherwise: Joi.valid('z') }),
b: Joi.any()
};
Alternatively, if you want to specify a specific type such as string
, array
, etc, you can do so like this:
const schema = {
a: Joi.valid('a', 'b', 'other'),
other: Joi.string()
.when('a', { is: 'other', then: Joi.required() }),
};
If you need to validate a child key inside a nested object based on a sibling's value, you can do so like this:
const schema = Joi.object().keys({
a: Joi.boolean().required(),
b: Joi.object()
.keys({
c: Joi.string(),
d: Joi.number().required()
})
.required()
.when('a', {
is: true,
then: Joi.object({ c: Joi.required() }) // b.c is required only when a is true
})
});
Overrides the key name in error messages.
name
- the name of the key.
const schema = {
first_name: Joi.string().label('First Name')
};
Outputs the original untouched value instead of the casted value.
isRaw
- whether to enable raw mode or not. Defaults to true.
const schema = {
timestamp: Joi.date().format('YYYYMMDD').raw()
};
Considers anything that matches the schema to be empty (undefined
).
schema
- any object or joi schema to match. An undefined schema unsets that rule.
let schema = Joi.string().empty('');
schema.validate(''); // returns { error: null, value: undefined }
schema = schema.empty();
schema.validate(''); // returns { error: "value" is not allowed to be empty, value: '' }
Overrides the default joi error with a custom error if the rule fails where:
err
- the override error.
Note that the provided error will be returned as-is, unmodified and undecorated with any of the
normal joi error properties. If validation fails and another error is found before the error
override, that error will be returned and the override will be ignored (unless the abortEarly
option has been set to false
).
let schema = Joi.string().error(new Error('Was REALLY expecting a string'));
schema.validate(3); // returns err.message === 'Was REALLY expecting a string'
Generates a schema object that matches an array data type. Note that undefined values inside arrays are not allowed by default but can be by using sparse()
.
Supports the same methods of the any()
type.
const array = Joi.array().items(Joi.string().valid('a', 'b'));
array.validate(['a', 'b', 'a'], (err, value) => { });
Allows this array to be sparse. enabled
can be used with a falsy value to go back to the default behavior.
let schema = Joi.array().sparse(); // undefined values are now allowed
schema = schema.sparse(false); // undefined values are now denied
Allows single values to be checked against rules as if it were provided as an array.
enabled
can be used with a falsy value to go back to the default behavior.
const schema = Joi.array().items(Joi.number()).single();
schema.validate([4]); // returns `{ error: null, value: [ 4 ] }`
schema.validate(4); // returns `{ error: null, value: [ 4 ] }`
Lists the types allowed for the array values where:
type
- a joi schema object to validate each array item against.type
can be an array of values, or multiple values can be passed as individual arguments.
If a given type is .required()
then there must be a matching item in the array.
If a type is .forbidden()
then it cannot appear in the array.
Required items can be added multiple times to signify that multiple items must be found.
Errors will contain the number of items that didn't match. Any unmatched item having a label will be mentioned explicitly.
const schema = Joi.array().items(Joi.string(), Joi.number()); // array may contain strings and numbers
const schema = Joi.array().items(Joi.string().required(), Joi.string().required()); // array must contain at least two strings
const schema = Joi.array().items(Joi.string().valid('not allowed').forbidden(), Joi.string()); // array may contain strings, but none of those strings can match 'not allowed'
const schema = Joi.array().items(Joi.string().label('My string').required(), Joi.number().required()); // If this fails it can result in `[ValidationError: "value" does not contain [My string] and 1 other required value(s)]`
Lists the types in sequence order for the array values where:
type
- a joi schema object to validate against each array item in sequence order.type
can be an array of values, or multiple values can be passed as individual arguments.
If a given type is .required()
then there must be a matching item with the same index position in the array.
Errors will contain the number of items that didn't match. Any unmatched item having a label will be mentioned explicitly.
const schema = Joi.array().ordered(Joi.string().required(), Joi.number().required()); // array must have first item as string and second item as number
const schema = Joi.array().ordered(Joi.string().required()).items(Joi.number().required()); // array must have first item as string and 1 or more subsequent items as number
const schema = Joi.array().ordered(Joi.string().required(), Joi.number()); // array must have first item as string and optionally second item as number
Specifies the minimum number of items in the array where:
limit
- the lowest number of array items allowed.
const schema = Joi.array().min(2);
Specifies the maximum number of items in the array where:
limit
- the highest number of array items allowed.
const schema = Joi.array().max(10);
Specifies the exact number of items in the array where:
limit
- the number of array items allowed.
const schema = Joi.array().length(5);
Requires the array values to be unique.
You can provide a custom comparator
function that takes 2 parameters to compare. This function should return whether the 2 parameters are equal or not, you are also responsible for this function not to fail, any Error
would bubble out of Joi.
Note: remember that if you provide a custom comparator, different types can be passed as parameter depending on the rules you set on items.
Be aware that a deep equality is performed on elements of the array having a type of object
, a performance penalty is to be expected for this kind of operation.
const schema = Joi.array().unique();
const schema = Joi.array().unique((a, b) => a.property === b.property);
Generates a schema object that matches a boolean data type (as well as the strings 'true', 'false', 'yes', 'no', 'on', 'off', 1, 0, '1', or '0'). Can also be called via bool()
.
Supports the same methods of the any()
type.
const boolean = Joi.boolean();
boolean.validate(true, (err, value) => { });
Generates a schema object that matches a Buffer data type (as well as the strings which will be converted to Buffers).
Supports the same methods of the any()
type.
const schema = Joi.binary();
Sets the string encoding format if a string input is converted to a buffer where:
encoding
- the encoding scheme.
const schema = Joi.binary().encoding('base64');
Specifies the minimum length of the buffer where:
limit
- the lowest size of the buffer.
const schema = Joi.binary().min(2);
Specifies the maximum length of the buffer where:
limit
- the highest size of the buffer.
const schema = Joi.binary().max(10);
Specifies the exact length of the buffer:
limit
- the size of buffer allowed.
const schema = Joi.binary().length(5);
Generates a schema object that matches a date type (as well as a JavaScript date string or number of milliseconds).
Supports the same methods of the any()
type.
const date = Joi.date();
date.validate('12-21-2012', (err, value) => { });
Specifies the oldest date allowed where:
date
- the oldest date allowed.
const schema = Joi.date().min('1-1-1974');
Notes: 'now'
can be passed in lieu of date
so as to always compare relatively to the current date, allowing to explicitly ensure a date is either in the past or in the future.
const schema = Joi.date().min('now');
It can also be a reference to another field.
const schema = Joi.object({
from: Joi.date().required(),
to: Joi.date().min(Joi.ref('from')).required()
});
Specifies the latest date allowed where:
date
- the latest date allowed.
const schema = Joi.date().max('12-31-2020');
Notes: 'now'
can be passed in lieu of date
so as to always compare relatively to the current date, allowing to explicitly ensure a date is either in the past or in the future.
const schema = Joi.date().max('now');
It can also be a reference to another field.
const schema = Joi.object({
from: Joi.date().max(Joi.ref('to')).required(),
to: Joi.date().required()
});
Specifies the allowed date format:
format
- string or array of strings that follow themoment.js
format.
const schema = Joi.date().format('YYYY/MM/DD');
Requires the string value to be in valid ISO 8601 date format.
const schema = Joi.date().iso();
Requires the value to be a timestamp interval from Unix Time.
type
- the type of timestamp (allowed values areunix
orjavascript
[default])
const schema = Joi.date().timestamp(); // defaults to javascript timestamp
const schema = Joi.date().timestamp('javascript'); // also, for javascript timestamp (milliseconds)
const schema = Joi.date().timestamp('unix'); // for unix timestamp (seconds)
Generates a schema object that matches a function type.
Supports the same methods of the object()
type. Note that validating a function keys will cause the function
to be cloned. While the function will retain its prototype and closure, it will lose its length
property value (will be
set to 0
).
const func = Joi.func();
func.validate(function () {}, (err, value) => { });
Specifies the arity of the function where:
n
- the arity expected.
const schema = Joi.func().arity(2);
Specifies the minimal arity of the function where:
n
- the minimal arity expected.
const schema = Joi.func().minArity(1);
Specifies the maximal arity of the function where:
n
- the maximum arity expected.
const schema = Joi.func().arity(3);
Requires the function to be a Joi reference.
const schema = Joi.func().ref();
Generates a schema object that matches a number data type (as well as strings that can be converted to numbers).
Infinity
and -Infinity
are invalid by default, you can change that behavior by calling allow(Infinity, -Infinity)
.
Supports the same methods of the any()
type.
const number = Joi.number();
number.validate(5, (err, value) => { });
Specifies the minimum value where:
limit
- the minimum value allowed.
const schema = Joi.number().min(2);
It can also be a reference to another field.
const schema = Joi.object({
min: Joi.number().required(),
max: Joi.number().min(Joi.ref('min')).required()
});
Specifies the maximum value where:
limit
- the maximum value allowed.
const schema = Joi.number().max(10);
It can also be a reference to another field.
const schema = Joi.object({
min: Joi.number().max(Joi.ref('max')).required(),
max: Joi.number().required()
});
Specifies that the value must be greater than limit
.
const schema = Joi.number().greater(5);
const schema = Joi.object({
min: Joi.number().required(),
max: Joi.number().greater(Joi.ref('min')).required()
});
Specifies that the value must be less than limit
.
const schema = Joi.number().less(10);
It can also be a reference to another field.
const schema = Joi.object({
min: Joi.number().less(Joi.ref('max')).required(),
max: Joi.number().required()
});
Requires the number to be an integer (no floating point).
const schema = Joi.number().integer();
Specifies the maximum number of decimal places where:
limit
- the maximum number of decimal places allowed.
const schema = Joi.number().precision(2);
Specifies that the value must be a multiple of base
:
const schema = Joi.number().multiple(3);
Notes: Joi.number.multiple(base)
uses the modulo operator (%) to determine if a number is multiple of another number.
Therefore, it has the normal limitations of Javascript modulo operator. The results with decimal/floats maybe incorrect.
Requires the number to be positive.
const schema = Joi.number().positive();
Requires the number to be negative.
const schema = Joi.number().negative();
Generates a schema object that matches an object data type (as well as JSON strings that parsed into objects). Defaults to allowing any child key.
Supports the same methods of the any()
type.
const object = Joi.object().keys({
a: Joi.number().min(1).max(10).integer(),
b: 'some string'
});
object.validate({ a: 5 }, (err, value) => { });
Sets or extends the allowed object keys where:
schema
- optional object where each key is assigned a joi type object. Ifschema
is{}
no keys allowed. Ifschema
isnull
orundefined
, any key allowed. Ifschema
is an object with keys, the keys are added to any previously defined keys (but narrows the selection if all keys previously allowed). Defaults to 'undefined' which allows any child key.
const base = Joi.object().keys({
a: Joi.number(),
b: Joi.string()
});
// Validate keys a, b and c.
const extended = base.keys({
c: Joi.boolean()
});
Notes: We have three different ways to define a schema for performing a validation
- Using the plain JS object notation:
const schema = {
a: Joi.string(),
b: Joi.number()
};
- Using the
Joi.object([schema])
notation
const schema = Joi.object({
a: Joi.string(),
b: Joi.number()
});
- Using the
Joi.object().keys([schema])
notation
const schema = Joi.object().keys({
a: Joi.string(),
b: Joi.number()
});
While all these three objects defined above will result in the same validation object, there are some differences in using one or another:
When using the {}
notation, you are just defining a plain JS object, which isn't a schema object.
You can pass it to the validation method but you can't call validate()
method of the object because it's just a plain JS object.
Besides, passing the {}
object to the validate()
method each time, will perform an expensive schema compilation operation on every validation.
Using Joi.object([schema])
will return a schema object, so you can call the validate()
method directly, e.g:
const schema = Joi.object({
a: Joi.boolean()
});
schema.validate(true, (err, value) => {
console.log('err: ', err);
});
When you use Joi.object([schema])
, it gets compiled the first time, so you can pass it to the validate()
method multiple times and no overhead is added.
Another benefits of using Joi.object([schema])
instead of a plain JS object is that you can set any options on the object like allowing unknown keys, e.g:
const schema = Joi.object({
arg: Joi.string().valid('firstname', 'lastname', 'title', 'company', 'jobtitle'),
value: Joi.string(),
}).pattern(/firstname|lastname/, Joi.string().min(2));
This is basically the same as Joi.object([schema])
, but using Joi.object().keys([schema])
is more useful when you want to add more keys (e.g. call keys()
multiple times). If you are only adding one set of keys, you can skip the keys()
method and just use object()
directly.
Some people like to use keys()
to make the code more explicit (this is style only).
Specifies the minimum number of keys in the object where:
limit
- the lowest number of keys allowed.
const schema = Joi.object().min(2);
Specifies the maximum number of keys in the object where:
limit
- the highest number of object keys allowed.
const schema = Joi.object().max(10);
Specifies the exact number of keys in the object where:
limit
- the number of object keys allowed.
const schema = Joi.object().length(5);
Specify validation rules for unknown keys matching a pattern where:
regex
- a regular expression tested against the unknown key names.schema
- the schema object matching keys must validate against.
const schema = Joi.object({
a: Joi.string()
}).pattern(/\w\d/, Joi.boolean());
Defines an all-or-nothing relationship between keys where if one of the peers is present, all of them are required as well where:
peers
- the key names of which if one present, all are required.peers
can be a single string value, an array of string values, or each peer provided as an argument.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).and('a', 'b');
Defines a relationship between keys where not all peers can be present at the same time where:
peers
- the key names of which if one present, the others may not all be present.peers
can be a single string value, an array of string values, or each peer provided as an argument.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).nand('a', 'b');
Defines a relationship between keys where one of the peers is required (and more than one is allowed) where:
peers
- the key names of which at least one must appear.peers
can be a single string value, an array of string values, or each peer provided as an argument.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).or('a', 'b');
Defines an exclusive relationship between a set of keys where one of them is required but not at the same time where:
peers
- the exclusive key names that must not appear together but where one of them is required.peers
can be a single string value, an array of string values, or each peer provided as an argument.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).xor('a', 'b');
Requires the presence of other keys whenever the specified key is present where:
key
- the reference key.peers
- the required peer key names that must appear together withkey
.peers
can be a single string value or an array of string values.
Note that unlike object.and()
, with()
creates a dependency only between the key
and each of the peers
, not
between the peers
themselves.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).with('a', 'b');
Forbids the presence of other keys whenever the specified is present where:
key
- the reference key.peers
- the forbidden peer key names that must not appear together withkey
.peers
can be a single string value or an array of string values.
const schema = Joi.object().keys({
a: Joi.any(),
b: Joi.any()
}).without('a', ['b']);
Renames a key to another name (deletes the renamed key) where:
from
- the original key name.to
- the new key name.options
- an optional object with the following optional keys:alias
- iftrue
, does not delete the old key name, keeping both the new and old keys in place. Defaults tofalse
.multiple
- iftrue
, allows renaming multiple keys to the same destination where the last rename wins. Defaults tofalse
.override
- iftrue
, allows renaming a key over an existing key. Defaults tofalse
.ignoreUndefined
- iftrue
, skip renaming of a key if it's undefined. Defaults tofalse
.
Keys are renamed before any other validation rules are applied.
const object = Joi.object().keys({
a: Joi.number()
}).rename('b', 'a');
object.validate({ b: 5 }, (err, value) => { });
Verifies an assertion where:
ref
- the key name or reference.schema
- the validation rules required to satisfy the assertion. If theschema
includes references, they are resolved against the object value, not the value of theref
target.message
- optional human-readable message used when the assertion fails. Defaults to 'failed to pass the assertion test'.
const schema = Joi.object().keys({
a: {
b: Joi.string(),
c: Joi.number()
},
d: {
e: Joi.any()
}
}).assert('d.e', Joi.ref('a.c'), 'equal to a.c');
Overrides the handling of unknown keys for the scope of the current object only (does not apply to children) where:
allow
- iffalse
, unknown keys are not allowed, otherwise unknown keys are ignored.
const schema = Joi.object({ a: Joi.any() }).unknown();
Requires the object to be an instance of a given constructor where:
constructor
- the constructor function that the object must be an instance of.name
- an alternate name to use in validation errors. This is useful when the constructor function does not have a name.
const schema = Joi.object().type(RegExp);
Requires the object to be a Joi schema instance.
const schema = Joi.object().schema();
Sets the specified children to required.
children
- can be a single string value, an array of string values, or each child provided as an argument.
const schema = Joi.object().keys({ a: { b: Joi.number() }, c: { d: Joi.string() } });
const requiredSchema = schema.requiredKeys('', 'a.b', 'c', 'c.d');
Note that in this example ''
means the current object, a
is not required but b
is, as well as c
and d
.
Sets the specified children to optional.
children
- can be a single string value, an array of string values, or each child provided as an argument.
const schema = Joi.object().keys({ a: { b: Joi.number().required() }, c: { d: Joi.string().required() } });
const requiredSchema = schema.optionalKeys('a.b', 'c.d');
The behavior is exactly the same as requiredKeys
.
Generates a schema object that matches a string data type. Note that empty strings are not allowed by default and must be enabled with allow('')
.
Supports the same methods of the any()
type.
const schema = Joi.string().min(1).max(10);
schema.validate('12345', (err, value) => { });
Allows the value to match any whitelist of blacklist item in a case insensitive comparison.
const schema = Joi.string().valid('a').insensitive();
Specifies the minimum number string characters where:
limit
- the minimum number of string characters required.encoding
- is specified, the string length is calculated in bytes using the provided encoding.
const schema = Joi.string().min(2);
It can also be a reference to another field.
const schema = Joi.object({
min: Joi.string().required(),
value: Joi.string().min(Joi.ref('min'), 'utf8').required()
});
Specifies the maximum number of string characters where:
limit
- the maximum number of string characters allowed.encoding
- if specified, the string length is calculated in bytes using the provided encoding.
const schema = Joi.string().max(10);
It can also be a reference to another field.
const schema = Joi.object({
max: Joi.string().required(),
value: Joi.string().max(Joi.ref('max'), 'utf8').required()
});
Specifies whether the string.max()
limit should be used as a truncation.
Parameters are:
enabled
- optional parameter defaulting totrue
which allows you to reset the behavior of truncate by providing a falsy value.
const schema = Joi.string().max(5).truncate();
Requires the number to be a credit card number (Using Lunh Algorithm).
const schema = Joi.string().creditCard();
Specifies the exact string length required where:
limit
- the required string length.encoding
- is specified, the string length is calculated in bytes using the provided encoding.
const schema = Joi.string().length(5);
It can also be a reference to another field.
const schema = Joi.object({
length: Joi.string().required(),
value: Joi.string().length(Joi.ref('length'), 'utf8').required()
});
Defines a regular expression rule where:
pattern
- a regular expression object the string value must match against.name
- optional name for patterns (useful with multiple patterns). Defaults to 'required'.
const schema = Joi.string().regex(/^[abc]+$/);
Replace characters matching the given pattern with the specified replacement string where:
pattern
- a regular expression object to match against, or a string of which all occurrences will be replaced.replacement
- the string that will replace the pattern.
const schema = Joi.string().replace(/b/gi, 'x');
schema.validate('abBc', (err, value) => {
// here value will be 'axxc'
});
When pattern
is a string all its occurrences will be replaced.
Requires the string value to only contain a-z, A-Z, and 0-9.
const schema = Joi.string().alphanum();
Requires the string value to only contain a-z, A-Z, 0-9, and underscore _.
const schema = Joi.string().token();
Requires the string value to be a valid email address.
options
- optional settings:errorLevel
- Numerical threshold at which an email address is considered invalid.tldWhitelist
- Specifies a list of acceptable TLDs.minDomainAtoms
- Number of atoms required for the domain. Be careful since some domains, such asio
, directly allow email.
const schema = Joi.string().email();
Requires the string value to be a valid ip address.
options
- optional settings:version
- One or more IP address versions to validate against. Valid values:ipv4
,ipv6
,ipvfuture
cidr
- Used to determine if a CIDR is allowed or not. Valid values:optional
,required
,forbidden
// Accept only ipv4 and ipv6 addresses with a CIDR
const schema = Joi.string().ip({
version: [
'ipv4',
'ipv6'
],
cidr: 'required'
});
Requires the string value to be a valid RFC 3986 URI.
options
- optional settings:scheme
- Specifies one or more acceptable Schemes, should only include the scheme name. Can be an Array or String (strings are automatically escaped for use in a Regular Expression).allowRelative
- Allow relative URIs. Defaults tofalse
.
// Accept git or git http/https
const schema = Joi.string().uri({
scheme: [
'git',
/git\+https?/
]
});
Requires the string value to be a valid GUID.
const schema = Joi.string().guid();
Requires the string value to be a valid hexadecimal string.
const schema = Joi.string().hex();
Requires the string value to be a valid hostname as per RFC1123.
const schema = Joi.string().hostname();
Requires the string value to be all lowercase. If the validation convert
option is on (enabled by default), the string
will be forced to lowercase.
const schema = Joi.string().lowercase();
Requires the string value to be all uppercase. If the validation convert
option is on (enabled by default), the string
will be forced to uppercase.
const schema = Joi.string().uppercase();
Requires the string value to contain no whitespace before or after. If the validation convert
option is on (enabled by
default), the string will be trimmed.
const schema = Joi.string().trim();
Requires the string value to be in valid ISO 8601 date format.
const schema = Joi.string().isoDate();
Generates a type that will match one of the provided alternative schemas via the try()
method. If no schemas are added, the type will not match any value except for undefined
.
Supports the same methods of the any()
type.
Alternatives can be expressed using the shorter []
notation.
const alt = Joi.alternatives().try(Joi.number(), Joi.string());
// Same as [Joi.number(), Joi.string()]
Adds an alternative schema type for attempting to match against the validated value where:
schema
- an array of alternative joi types. Also supports providing each type as a separate argument.
const alt = Joi.alternatives().try(Joi.number(), Joi.string());
alt.validate('a', (err, value) => { });
Adds a conditional alternative schema type based on another key (not the same as any.when()
) value where:
ref
- the key name or reference.options
- an object with:is
- the required condition joi type.then
- the alternative schema type to try if the condition is true. Required ifotherwise
is missing.otherwise
- the alternative schema type to try if the condition is false. Required ifthen
is missing.
const schema = {
a: Joi.alternatives().when('b', { is: 5, then: Joi.string(), otherwise: Joi.number() }),
b: Joi.any()
};
Note that when()
only adds additional alternatives to try and does not impact the overall type. Setting
a required()
rule on a single alternative will not apply to the overall key. For example,
this definition of a
:
const schema = {
a: Joi.alternatives().when('b', { is: true, then: Joi.required() }),
b: Joi.boolean()
};
Does not turn a
into a required key when b
is true
. Instead, it tells the validator to try and match the
value to anything that's not undefined
. However, since Joi.alternatives()
by itself allows undefined
, the rule
does not accomplish turning a
to a required value. This rule is the same as Joi.alternatives([Joi.required()])
when b
is true
which will allow any value including undefined
.
To accomplish the desired result above use:
const schema = {
a: Joi.when('b', { is: true, then: Joi.required() }),
b: Joi.boolean()
};
Generates a placeholder schema for a schema that you would provide with the fn
.
Supports the same methods of the any()
type.
This is mostly useful for recursive schemas, like :
const Person = Joi.object({
firstName: Joi.string().required(),
lastName: Joi.string().required(),
children: Joi.array().items(Joi.lazy(() => Person).description('Person schema'))
});
Joi throws classical javascript Error
s containing :
name
-'ValidationError'
.isJoi
-true
.details
- an array of errors :message
- string with a description of the error.path
- dotted path to the key where the error happened.type
- type of the error.context
- object providing context of the error.
annotate
- function that returns a string with an annotated version of the object pointing at the places where errors occured._object
- the original object to validate.