Skip to content

trex-arms/body_validator

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

20 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

body_validator

Designed for validating HTTP POST bodies in a TypeScript-friendly way

Install

Not currently on npm. Download the files from Github. Sorry!

Example

const v = require('body_validator')
// or
import v from 'body_validator'

Valid example

const validator = v.object({
	name: v.string,
	age: v.number,
	cool: v.optional(v.boolean),
	powers: v.array(v.string),
})

/** @type {unknown} */
const valid_input = {
	name: 'Batman',
	age: 27,
	cool: true,
	powers: [ 'money' ],
}

if (validator.is_valid(valid_input)) {
	// Should pass the type system
	valid_input.name // => 'Batman'
}

validator.get_messages(valid_input, `input`) // => []

Invalid example

/** @type {unknown} */
const invalid_input = {
	name: 'Superman',
}

validator.is_valid(invalid_input) // => false

validator.get_messages(invalid_input, `input`) // => [ '"input"."age" is not a number', '"input"."powers" is not an array' ]

API

import v from 'body_validator'

Validator object type

A Validator object has the following methods:

  • is_valid(input): boolean is a Type Predicate function that returns true/false for whether the input is that type
  • get_messages(input): string[] is a function that returns an array of string error messages

v.boolean

v.boolean.is_valid(true) // => true
v.boolean.is_valid(false) // => true
v.boolean.is_valid('no') // => false
v.boolean.get_messages(true, 'test') // => []
v.boolean.get_messages('no', 'test') // => [ '"test" is not a boolean' ]

v.string

v.string.is_valid(':)') // => true
v.string.is_valid(false) // => false

v.number

v.number.is_valid(3) // => true
v.number.is_valid(-17.32) // => true
v.number.is_valid('no') // => false

v.integer

v.integer.is_valid(3) // => true
v.integer.is_valid(3.14159) // => false
v.integer.is_valid({}) // => false

v.null

v.null.is_valid(null) // => true
v.null.is_valid(undefined) // => false

v.date

v.date.is_valid(new Date()) // => true
v.date.is_valid('2024-07-18T14:53:55.123Z') // => false

v.exact(value)

v.exact('hi').is_valid('hi') // => true
v.exact('hi').is_valid('HI') // => false
v.exact({}).is_valid({}) // => false
const x = {}
v.exact(x).is_valid(x) // => true

v.array(validator: Validator)

(No passing in v.optional.)

v.array(v.boolean).is_valid([]) // => true
v.array(v.boolean).is_valid([ true ]) // => true
v.array(v.boolean).is_valid(true) // => false
v.array(v.boolean).is_valid([ 'WRONG' ]) // => false

v.object(object_of_validators: Record<string, Validator>)

const cool_validator = v.object({ cool: v.boolean })

cool_validator.is_valid({ cool: true }) // => true

cool_validator.is_valid({}) // => false
cool_validator.is_valid({ cool: 'no' }) // => false
cool_validator.is_valid({ uncool: true }) // => false

v.object_values(validator: Validator)

const str_values_validator = v.object_values(v.string)

str_values_validator.is_valid({ hello: 'world' }) // => true
str_values_validator.is_valid({ non_string: 3 }) // => false
str_values_validator.is_valid({ hello: 'darkness', non_string: 3 }) // => false
str_values_validator.is_valid({}) // => true

v.one_of(...validators: Validator[])

const enum_validator = v.one_of(v.exact('GET'), v.exact('POST'), v.exact('DELETE'))

enum_validator.is_valid('GET') // => true
enum_validator.is_valid('PATCH') // => false

v.regex(re: RegExp)

const enum_validator2 = v.regex(/^(GET|POST|DELETE)$/)

enum_validator2.is_valid('POST') // => true
enum_validator2.is_valid('OPTIONS') // => false

v.nullable(validator: Validator)

v.boolean.is_valid(null) // => false
v.nullable(v.boolean).is_valid(null) // => true
v.nullable(v.boolean).is_valid(false) // => true

v.optional(validator: Validator)

v.boolean.is_valid(undefined) // => false
v.optional(v.boolean).is_valid(undefined) // => true
v.optional(v.boolean).is_valid(false) // => true

This is intended for use with v.object.

const options_validator = v.object({
	name: v.string,
	age: v.optional(v.integer),
})

options_validator.is_valid({ name: 'Joseph' }) // => true
options_validator.is_valid({ name: 'Joseph', age: 42 }) // => true
options_validator.is_valid({ name: 'Joseph', age: undefined }) // => true
options_validator.is_valid({ age: 42 }) // => false

v.custom(options)

options is an object with these properties:

  • function is_valid(input: any) => boolean
  • function get_messages(input: any, name: string) => string[] The resulting array should be empty if is_valid would return true for that input, but should have at least one string if is_valid would return false for that input
// Example: IP Address validator
function is_valid(input) {
	if (typeof input !== 'string') return false

	const parts = input.split('.')
	if (parts.length !== 4) return false

	return parts
		.map(p => Number(p))
		.every(n => !Number.isNaN(n) && n < 256)
}

function get_messages(input, name) {
	return is_valid(input)
		? []
		: [ `"${ name }" is not a valid IP Address` ]
}

const ip_validator = v.custom({ is_valid, get_messages })

ip_validator.is_valid('192.168.0.1') // => true
ip_validator.is_valid('1.1.1.1') // => true

ip_validator.is_valid('256.1.2.3') // => false
ip_validator.is_valid('1.2.3') // => false
ip_validator.is_valid('1.1.1.1a') // => false

License

MIT

About

No description, website, or topics provided.

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •