Uvo
Uvo is a javascript universal validation library. The main goal is to provide schema creation tool for convenient and flexible form's validation with the only one call. Schema can represent either simple form with few fields or huge one with a big amount of cross form validations, recursive descents, different conditions and custom data processing.
Uvo wants to be a pretty small size library, so each validator represented as separated module and can be skipped during treeshaking stage in your bundler (e.g. webpack
). Only templating API
represented as one indivisible bundle.
Uvo wants to be a flexible and comprehensive library, so uvo/extended
will extend base api with a huge base of specific validators such as email
, url
, guid
and so on. Also uvo supports asynchronous validations.
Bundles (minified) | ESM | CJS | UMD |
---|---|---|---|
Base API | ~6.8kb | ~6.8kb | ~6.5kb |
Templating API | ~9.7kb | ~9.8kb | ~9.9kb |
Extended API | ~2.9kb | ~3kb | ~2.9kb |
Extended Template API | ~4.8kb | ~5kb | ~5kb |
Uvo has own types definition file for typescript
.
Installation
Install with npm:
npm install uvo
Install with yarn:
yarn add uvo
Usage
First of all let's import it:
;;
or
const v = ;const e = ; // es5const v = ;const e = ;
Each validator returns initial or converted value on success, otherwise null
.
Single validator usage:
v 10 ; // => 10v '12.5' ; // => 12.5v 'str' ; // => null
Lets group them in a sequence:
v 10 ; // => 10v -1 ; // => nullv 'str' ; // => null
Simple form creation:
vobject id: v v v name: v v v v roles: v v roles ;
In the example above you can see not only validators (namely trim
). Library has processor
modules which are only transform value into another. You can inject your custom processors and validators as we did for roles filtration.
Previous example use classic object scheme. It won't give us great flexibility in the future. Lets write new one with advanced object scheme:
const now = Date; v;
Advanced scheme gives us some new features such as keys aggregation
and multiple stages for keys
.
Let's do this:
v;
Now we will always use object2
scheme as recommended by default.
The next step is about cross form validations. For this issue we must use containers
.
The most important containers: withMeta
, withErrors
, withPromise
.
v;
In the above schema we use setRef
and getRef
(i.e. referencing) to compare fields to each other.
In that case we must use withMeta
container. It provides global data storage for scheme, allows us to use references and stores meta data for errors.
What about errors? Each validator takes an additional parameter for error. Me must use withErrors
container for errors accumulation from scheme.
const scheme = v; ; // => { result: 10.5, errors: null }; // => { result: 10, errors: ['err3'] }; // => { result: 10, errors: ['err2', 'err3'] }; // => { result: 10, errors: ['err1'] }
You can provide not only literal for error, but function. Function will be called with meta data (if withMeta
provided) which contains path
and validator
for occured error. Grouper parallel
provides nonsequential validation and you can collect all errors from it. Also you can use or
grouper, if value type can be different.
We have overviewed only basic features. For more information see API documentation.
Async API
Uvo provides flexible asynchronous validations. withPromise
container makes entire scheme async.
const validator = v; ;// => { id: 1, name: 'name' } // More promises;// => { id: 1, name: null }
All promises will be awaited implicitly in the scheme. If you want to catch errors or provide custom ones for promises, make sure to use withErrors
container and async
validator.
withPromise
is a low level container, use it inside withErrors
and withMeta
containers.
v;
One more important feature is cross promise validations with async
and wait
.
withMeta
is required.
v;
Templates
Library also has own templating API for advanced usage. Compiler for template minification in progress. Let's rewrite all examples from above:
; // base version /* injections for template (object or array) */ /* errors for template (object or array) */ ; // short versiontml`...` /* injections for template (object or array) */ /* errors for template (object or array) */ ; // base number 10 ; // => 10 // short numbertml`@n` 'str' ; // => null
To provide validators from uvo/extended
use provide
function and builders from uvo/extended-template
.
;; ; ; // or provide all. Provides with the same name as in the 'uvo/extended'. ; ;
Templates has own universal comparator @compare
. It replaces such validators as gte
, oneOf
, regex
and so on.
// long sequence 10 ; // => 10 // short sequencetml`@n @c( >=0 )` NaN ; // => null
Literal $
marks injections.
Our "simple form":
name // template does not have such processor by itself roleRegEx: /^[A-Z]{5,20}$/ uniqueVal: null // `unique` validator for template in progress roles; // short versiontml` @o( id @n @c( >=0, %1 ), name @s $0 @l( >=8, <=24 ), roles @a( @s @c( *$1 ) ) $2 $3 )` name /^[A-Z]{5,20}$/ null roles; 'createdAt' 'modifiedAt' 'deletedAt' /.+/ date;
Groupers in templates have own syntax (e.g. parallel
- <{ }>
).
Literal #...
sets current value into reference. If name is not provided, the latest path node will be used from meta
.
Error can be provided via !
literal after validator.
Cross form validations and errors:
Date modifiedAt + 60 * 1000; // short onetml` @o( createdAt @d @c( >=$0 ) #0, modifiedAt @d @c( >=#0 ) #1, deletedAt @d @c( >=$1( #1 ) ) ) ~m` Date modifiedAt + 60 * 1000; null err1: 'err1' err2: 'err2' err3: 'err3' ; // shorttml` @n!0 <{ @c( <=100 )!1 @c( !%1 )!2 }> ~e` null 'err1' 'err2' 'err3';
Async API in deed:
; // ValidatorErrors null 'promiseErr'; // Cross promise validation { /* do something asynchronous m.b. */ } ;
API
Base API | Templating API | Description |
---|---|---|
array | @array(...) @a(...) |
Checks value to be an array. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
async | @async(...) @p(...) |
Settles value to async storage. Can be awaited somewhere later. Type: validator. Returns input value on success. |
bool | @bool @b |
Checks value to be a boolean compatible. Converts on success. Use bool from Extended API for check only. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
date | @date @d |
Checks value to be a date compatible. Result in ms. Converts on success. Use date from Extended API for check only. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
fields | Checks for fields in the input object. Type: validator. Returns input value on success. | |
is | injected function via $... |
Checks value with custom comparator. Type: validator. Returns input value on success. |
length | @length(=...) @l(=...) @length(!=...) @l(!=...) |
Compares length with 'len' param. Requires to be an object like or string. Type: validator. Returns input value on success. |
multiple | @compare(%...) @c(%...) @compare(!%...) @c(!%...) |
Checks number to be an integer. Type: validator. Returns input value on success. |
number | @number @n |
Checks value to be a number compatible. Converts on success. Use number from Extended API for check only. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
object | @object(...) @o(...) |
Checks value to be an object. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
object2 | @object(...) @o(...) |
Checks value to be an object. Provides strict ordering. Each key can be a Regex. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
string | @string @s |
Checks value to be a string compatible. Converts on success. Use string from Extended API for check only. Type: semi validator, semi processor. If validation is successful, then converts value to proper type. |
keysMap | Maps object keys with custom mapper. Type: processor. Processor does not check params' and values' types. Escape usage without validators. | |
strip | Removes field from object conditionally. Type: processor. Processor does not check params' and values' types. Escape usage without validators. | |
valueMap | Maps value with custom mappers. Type: processor. Processor does not check params' and values' types. Escape usage without validators. | |
consecutive | <( ... )> |
Groups validators sequentially. Passes value through a sequence of validators until an error occurs. Uses by default in 'object' and 'object2' validator's scheme for fields. Type: grouper. Groups validators into one. |
or | <[ ... ]> |
Groups validators sequentially. Searches for first successful validator's result. Type: grouper. Groups validators into one. |
parallel | <{ ... }> |
Groups validators in parallel. The main goal is to catch all errors (pass value through a sequence of validators, even if an error occurred somewhere). Beware of using processors inside. Type: grouper. Groups validators into one. |
transform | Groups processors sequentially. Passes value through a sequence of processors. Takes only processors (doesn't check errors). Type: grouper. Groups validators into one. | |
withErrors | ~error(...) ~e(...) |
Provides error handling mechanism. Type: container. Embraces validators. Provides additional input processing. |
withFallback | @fallback(...) @f(...) |
Provides fallback value on error. Type: container. Embraces validators. Provides additional input processing. |
withMeta | ~meta(...) ~m(...) |
Provides meta structure. Can catch scheme logs. Type: container. Embraces validators. Provides additional input processing. |
withOnError | Provides custom error handler. Type: container. Embraces validators. Provides additional input processing. | |
withPromise | ~promise ~p |
Convert result to promise. Use it for async validation. Type: container. Embraces validators. Provides additional input processing. |
dynamic | conditional validation via ? or injection via $... |
Inserts new validators into scheme dynamically. Type: spreader. Spreads data through a validators scheme. |
getRef | as parameter via #... or as validators via ##... |
Takes value from spreaded structure. Might be used for dynamic validators creation. If 'preValidator' not provided, just replaces current value. Works only with provided meta object. Type: spreader. Spreads data through a validators scheme. |
setRef | #... |
Puts value into spreaded structure. If 'extValue' is provided, puts it instead of current value. i.e. reference api. Type: spreader. Spreads data through a validators scheme. |
setVRef | #...(...) |
Puts validators into spreaded structure. Might be used for recursive schemes. Type: spreader. Spreads data through a validators scheme. |
useDefault | @default(...) |
Puts default value into spreaded structure. If input value is empty, puts default value instead, otherwise validates input values with provided validators. If you need fallback value on error use 'withFallback' container instead. Type: spreader. Spreads data through a validators scheme. |
wait | @wait(...) @w(...) |
Waits for specified promise. Type: spreader. Spreads data through a validators scheme. |
Extended API
Base API | Templating API | Description |
---|---|---|
alpha | via provide |
Checks if the string contains only letters (a-zA-Z). Type: validator. Returns input value on success. |
alphanum | via provide |
Checks if the string contains only letters (a-zA-Z) and numbers. Type: validator. Returns input value on success. |
bin | via provide |
Checks if the string is a binary number. Type: validator. Returns input value on success. |
bool | via provide |
Checks for boolean type. Type: validator. Returns input value on success. |
contains | via provide |
Checks if the string or array contains the seed. Type: validator. Returns input value on success. |
date | via provide |
Checks for right date. Type: validator. Returns input value on success. |
emailFast | via provide |
Fast email validation. Type: validator. Returns input value on success. |
via provide |
Email validation. Type: validator. Returns input value on success. | |
even | via provide |
Checks number to be an even one. Type: validator. Returns input value on success. |
float | via provide |
Checks number to be float. Type: validator. Returns input value on success. |
hex | via provide |
Checks if the string is a hexadecimal number. Type: validator. Returns input value on success. |
integer | via provide |
Checks number to be an integer. Type: validator. Returns input value on success. |
lowercase | via provide |
Checks string to be in a lower case. Type: validator. Returns input value on success. |
negative | via provide |
Checks number to be negative. Type: validator. Returns input value on success. |
notContains | via provide |
Checks if the string or array does not contain the seed. Type: validator. Returns input value on success. |
number | via provide |
Checks for number type. Type: validator. Returns input value on success. |
oct | via provide |
Checks if the string is a octal number. Type: validator. Returns input value on success. |
positive | via provide |
Checks number to be positive. Type: validator. Returns input value on success. |
string | via provide |
Checks for string type. Type: validator. Returns input value on success. |
unique | via provide |
Checks array's elements to be unique. Type: validator. Returns input value on success. |
uppercase | via provide |
Checks string to be in an upper case. Type: validator. Returns input value on success. |
url | via provide |
URL validation. Type: validator. Returns input value on success. |
uuid | via provide |
UUID validation. Type: validator. Returns input value on success. |
clamp | via provide |
Clamps value to required boundaries. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
erase | via provide |
Erase input. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
repeat | via provide |
Repeats the string. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
round | via provide |
Round input number with specific method. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
toLower | via provide |
Lowercase input string. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
toUpper | via provide |
Uppercase input string. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
trim | via provide |
Trim input string with specific method. Type: processor. Processor does not check params' and values' types. Escape usage without validators. |
Performance
Uvo is a pretty fast library for web apps. uvo/template
will be dramatically optimized for BE usage in the future.
Tested with this benchmark.
Platform info:
==============
Windows_NT 10.0.18362 x64
Node.JS: 12.16.1
V8: 7.8.279.23-node.31
Intel(R) Core(TM) i5-3317U CPU @ 1.70GHz × 4
Suite: Simple object
√ validator.js 253,025 rps
√ validate.js 101,932 rps
√ validatorjs 53,471 rps
√ joi 25,926 rps
√ ajv 2,100,936 rps
√ mschema 198,256 rps
√ parambulator 4,452 rps
√ fastest-validator 2,096,706 rps
√ yup 5,903 rps
√ uvo 300,253 rps
validator.js -87.96% (253,025 rps) (avg: 3μs)
validate.js -95.15% (101,932 rps) (avg: 9μs)
validatorjs -97.45% (53,471 rps) (avg: 18μs)
joi -98.77% (25,926 rps) (avg: 38μs)
ajv 0% (2,100,936 rps) (avg: 475ns)
mschema -90.56% (198,256 rps) (avg: 5μs)
parambulator -99.79% (4,452 rps) (avg: 224μs)
fastest-validator -0.2% (2,096,706 rps) (avg: 476ns)
yup -99.72% (5,903 rps) (avg: 169μs)
uvo -85.71% (300,253 rps) (avg: 3μs)
-----------------------------------------------------------------------