Mikamin lightweight schema builder for value parsing and synchronous validation.
# Install with pnpm
pnpm add mikamin
# Install with npm
npm install mikamin
You define and create schema objects. Schema objects are immutable, so each call of a method returns a new schema object. Basic schema definition:
import {
bool,
Infer,
oneOf,
string,
optional,
nullable,
stringNumber,
handleSchema,
} from 'mikamin'
enum Gender {
Woman,
Man,
}
type Search = Infer<typeof searchSchema>
const searchSchema = {
ids: list(string),
genders: list(oneOf(Gender)),
cityId: nullable(string),
addiction: optional(stringNumber),
age: {
from: stringNumber,
to: stringNumber,
},
hasPhoto: bool,
}
// check validity
const errors = handleSchema({
schema: searchSchema,
values: {
genders: [1],
cityId: '327',
},
})
const hasError = Object.keys(errors).length != 0
console.log(errors)
// => {
// age: { from: 'from must be string!', to: 'to must be string!' },
// hasPhoto: 'hasPhoto must be boolean!'
// }
When you call handleSchema
, he have parametr fromatter
, and you able to detect user language, and after just
const formatErrorMsg = (lang: Language): ErrorFormater => ({ rule, name, meta }) => {
const dictionary = dictionaries[lang]
switch (rule) {
case 'string':
return dictionary.STRING.replace('#name#', name)
case 'requiredList':
return dictionary.REQUIRED_LIST.replace('#name#', name)
case 'optional':
return dictionary.OPTIONAL.replace('#name#', name)
case 'array':
return dictionary.ARRAY.replace('#name#', name)
case 'email':
return dictionary.EMAIL.replace('#name#', name)
case 'number':
return dictionary.NUMBER.replace('#name#', name)
case 'oneOf':
return dictionary.ONE_OF
.replace('#name#', name)
.replace('#meta#', meta!.values.join(', '))
default:
return dictionary.UNKNOWN
}
}
// Validation
const errors = handleSchema({
schema,
values: req.body,
formater: formatErrorMsg(req.lang),
})
type Validate<T> = preHandlerHookHandler<RawServerDefault, RawRequestDefaultExpression, RawReplyDefaultExpression, {Body: T}>
export const mikaminHandler = (schema: any): Validate<any> => (req, res, done) => {
req.lang = parseAcceptLanguage(req.headers['accept-language']!)
const errors = handleSchema({
schema,
values: req.body,
formater: formatErrorMsg(req.lang),
})
if (Object.keys(errors).length != 0) {
return res.status(400).send({errors})
}
done()
}
And use it in handler:
// handler.ts
import * as input from './input'
fastify.route<{Body: input.Register}>({
url: '/register',
method: 'POST',
preHandler: mikaminHandler(input.registerSchema),
handler: req => services.AuthService.register(req.body),
})
// input.ts
export type Register = Infer<typeof registerSchema>
export const registerSchema = {
username: string,
email: email,
password: string,
}