Antity.js is an Open source library for easy entity management.
- 🪶 Very lightweight
- đź§Ş Thoroughly tested
- đźšš Shipped as EcmaScrypt module
- 📝 Written in Typescript
- node: 22
This is the oldest targeted versions. The library should work properly on older versions of Node.js but we do not support it officially.
$ npm i @dwtechs/antityimport { Entity } from "@dwtechs/antity";
import { normalizeName, normalizeNickname } from "@dwtechs/checkard";
const entity = new Entity("consumers", [
{
key: "id",
type: "integer",
min: 0,
max: 120,
typeCheck: true,
methods: ["GET", "PUT", "DELETE"],
required: true,
safe: true,
sanitize: true,
normalize: true,
validate: true,
sanitizer: null,
normalizer: null,
validator: null,
},
{
key: "firstName",
type: "string",
min: 0,
max: 255,
typeCheck: true,
methods: ["GET", "POST", "PUT", "DELETE"],
required: true,
safe: true,
sanitize: true,
normalize: true,
validate: true,
sanitizer: null,
normalizer: normalizeName,
validator: null,
},
{
key: "lastName",
type: "string",
min: 0,
max: 255,
typeCheck: true,
methods: ["GET", "POST", "PUT", "DELETE"],
required: true,
safe: true,
sanitize: true,
normalize: true,
validate: true,
sanitizer: null,
normalizer: normalizeName,
validator: null,
},
{
key: "nickname",
type: "string",
min: 0,
max: 255,
typeCheck: true,
methods: ["GET", "POST", "PUT", "DELETE"],
required: true,
safe: true,
sanitize: true,
normalize: true,
validate: true,
sanitizer: null,
normalizer: normalizeNickname,
validator: null,
},
]);
// add a consumer. Used when logging in from user service
router.post("/", entity.normalizeArray, entity.validateArray, ...);
// or use check method to normalize and validate array at once
router.put("/", entity.check, ...);type Type =
"boolean" |
"string" |
"number" |
"integer" |
"float" |
"even" |
"odd" |
"positive" |
"negative" |
"powerOfTwo" |
"ascii" |
"array" |
"jwt" |
"symbol" |
"password" |
"email" |
"regex" |
"json" |
"ipAddress" |
"slug" |
"hexadecimal" |
"date" |
"timestamp" |
"function" |
"htmlElement" |
"htmlEventAttribute" |
"node" |
"object";
type Method = "GET" | "PATCH" | "PUT" | "POST" | "DELETE";
class Property {
key: string;
type: Type;
min: number | Date | null;
max: number | Date | null;
required: boolean;
safe: boolean;
typeCheck: boolean;
methods: Method[];
sanitize: boolean;
normalize: boolean;
validate: boolean;
sanitizer: Function | null;
normalizer: Function | null;
validator: Function | null;
};
class Entity {
constructor(name: string, properties: Property[]);
get name(): string;
get unsafeProps(): string[];
get properties(): Property[];
set name(name: string);
/**
* Returns a single property object matching the given key.
*
* - Searches the entity's properties for a property with the specified key
* - Useful for dynamic validation, normalization, or documentation
*
* @param {string} key - The property key to look up
* @returns {Property | undefined} The Property object if found, otherwise undefined
*
* **Input Properties Required:**
* - `key` (string) - Property key to look up
*
* **Output Properties:**
* - Property object matching the key, or undefined if not found
*
* @example
* ```typescript
* const prop = entity.getProp('firstName');
* // prop contains the Property object for 'firstName' or undefined
* ```
*/
getProp(key: string): Property | undefined;
/**
* Returns all properties configured for a given REST method.
*
* - Filters the entity's properties by the specified method (e.g., 'POST', 'GET')
* - Useful for dynamic validation, normalization, or documentation
*
* @param {Method} method - The REST method to filter properties by (e.g., 'POST', 'GET')
* @returns {Property[]} Array of Property objects associated with the method
*
* **Input Properties Required:**
* - `method` (string) - REST method to filter by
*
* **Output Properties:**
* - Array of Property objects matching the method
*
* @example
* ```typescript
* const postProps = entity.getPropsByMethod('POST');
* // postProps contains all properties relevant for POST requests
* ```
*/
getPropsByMethod(method: Method): Property[];
/**
* Normalizes an array of records by applying sanitization and normalization
* rules defined in the properties of the class.
*
* - Applies sanitization if `sanitize: true`
* - Applies normalization if `normalize: true`
* - Mutates req.body.rows or req.body with sanitized/normalized values
* - Calls next(error) on failure, next() on success
*
* @param {Request} req - Express request object containing rows
* @param {Response} _res - Express response object (not used)
* @param {NextFunction} next - Express next function
*
* @returns {void}
*
* **Input Properties Required:**
* - `req.body.rows` (array) or `req.body` (array) - Array of objects to normalize
* - Each property config can specify sanitize, normalize, etc.
*
* **Output Properties:**
* - Mutates array with sanitized/normalized values
* - Calls next(error) if normalization fails, next() if all pass
*
* @example
* ```typescript
* router.post('/entities', entity.normalizeArray, (req, res) => {
* // req.body.rows are now sanitized and normalized
* res.json({ success: true });
* });
* ```
*/
normalizeArray: (req: Request, _res: Response, next: NextFunction) => void;
/**
* Normalizes a single record by applying sanitization and normalization
* rules defined in the properties of the class.
*
* - Applies sanitization if `sanitize: true`
* - Applies normalization if `normalize: true`
* - Mutates req.body with sanitized/normalized values
* - Calls next(error) on failure, next() on success
*
* @param {Request} req - Express request object containing a single record
* @param {Response} _res - Express response object (not used)
* @param {NextFunction} next - Express next function
*
* @returns {void}
*
* **Input Properties Required:**
* - `req.body` (object) - Single object to normalize
* - Each property config can specify sanitize, normalize, etc.
*
* **Output Properties:**
* - Mutates `req.body` with sanitized/normalized values
* - Calls next(error) if normalization fails, next() if success
*
* @example
* ```typescript
* router.post('/entity', entity.normalizeOne, (req, res) => {
* // req.body is now sanitized and normalized
* res.json({ success: true });
* });
* ```
*/
normalizeOne: (req: Request, _res: Response, next: NextFunction) => void;
/**
* Validates an array of rows according to property config and HTTP method.
*
* - Checks required properties and validates values
* - Calls next(error) on failure, next() on success
*
* @param {Request} req - Express request object containing rows
* @param {Response} _res - Express response object (not used)
* @param {NextFunction} next - Express next function
*
* @returns {void}
*
* **Input Properties Required:**
* - `req.body.rows` (array) or `req.body` (array) - Array of objects to validate
* - Each property config can specify validate, required, etc.
*
* **Output Properties:**
* - Calls next(error) if any row fails validation, next() if all pass
*
* @example
* ```typescript
* router.post('/entities', entity.validateArray, (req, res) => {
* // req.body.rows are now validated
* res.json({ success: true });
* });
* ```
*/
validateArray: (req: Request, _res: Response, next: NextFunction) => void;
/**
* Validates a single record according to property config and HTTP method.
*
* - Checks required properties and validates values
* - Calls next(error) on failure, next() on success
*
* @param {Request} req - Express request object containing a single record
* @param {Response} _res - Express response object (not used)
* @param {NextFunction} next - Express next function
*
* @returns {void}
*
* **Input Properties Required:**
* - `req.body` (object) - Single object to validate
* - Each property config can specify validate, required, etc.
*
* **Output Properties:**
* - Calls next(error) if validation fails, next() if success
*
* @example
* ```typescript
* router.post('/entity', entity.validateOne, (req, res) => {
* // req.body is now validated
* res.json({ success: true });
* });
* ```
*/
validateOne: (req: Request, _res: Response, next: NextFunction) => void;
/**
* Checks, sanitizes, normalizes, and validates each row in req.body.rows according to property config and HTTP method.
*
* - Applies sanitization if `sanitize: true` and method matches
* - Applies normalization if `normalize: true` and method matches
* - Checks required properties and validates values
* - Calls next(error) on failure, next() on success
*
* @param {Request} req - Express request object containing rows
* @param {Response} _res - Express response object (not used)
* @param {NextFunction} next - Express next function
*
* @returns {void}
*
* **Input Properties Required:**
* - `req.body.rows` (array) - Array of objects to check
* - Each property config can specify sanitize, normalize, validate, required, etc.
*
* **Output Properties:**
* - Mutates `req.body.rows` with sanitized/normalized values
* - Calls next(error) if any row fails checks, next() if all pass
*
* @example
* ```typescript
* router.post('/entity', entity.check, (req, res) => {
* // req.body.rows are now sanitized, normalized, and validated
* res.json({ success: true });
* });
* ```
*/
check: (req: Request, _res: Response, next: NextFunction) => void;
}normalizeArray(), normalizeOne(), validateArray(), and validateOne() methods are made to be used as Express.js middlewares.
- normalizeArray() and validateArray() will look for data in the req.body.rows parameter or req.body as an array.
- normalizeOne() and validateOne() will look for data in the req.body parameter as a single object.
Password validation will have the following options by default :
const PWD_MIN_LENGTH = 9;
const PWD_MAX_LENGTH = 20;
const PWD_NUMBERS = true; // password must contain a number
const PWD_UPPERCASE = true; // password must contain an uppercase letter
const PWD_LOWERCASE = true; // password must contain a lowercase letter
const PWD_SYMBOLS = true; // password must contain at least one of the following symbol character : !@#%*_-+=:?><./()You can update password default validator by setting the following environment variables :
PWD_MIN_LENGTH_POLICY,
PWD_MAX_LENGTH_POLICY,
PWD_NUMBERS_POLICY,
PWD_UPPERCASE_POLICY,
PWD_LOWERCASE_POLICY,
PWD_SYMBOLS_POLICYProperties min and max of the password properties will override default and environement variable if set.
Any of these can be passed into the options object for each function.
| Name | Type | Description | Default value |
|---|---|---|---|
| key | string | Name of the property | |
| type | Type | Type of the property | |
| min | number | Date | Minimum value | 0 | 1900-01-01 |
| max | number | Date | Maximum value | 999999999 | 2200-12-31 |
| required | boolean | Property is required during validation | false |
| safe | boolean | Property is sent in the response | true |
| typeCheck | boolean | Type is checked during validation | false |
| methods | Method[] | property is validated for the listed methods only | [ "GET", "POST", "PUT", "DELETE" ] |
| sanitize | boolean | Sanitize the property if true | true |
| normalize | boolean | Normalize the property if true | false |
| validate | boolean | validate the property if true | true |
| sanitizer | ((v:any) => any) | null | Custom sanitizer function if sanitize is true | null |
| normalizer | ((v:any) => any) | null | Custop Normalizer function if normalize is true | null |
| validator | ((v:any, min:number, max:number, typeCheck:boolean) => any) | null | validator function if validate is true | null |
- Min and max parameters are not used for boolean type
- TypeCheck Parameter is not used for boolean, string and array types
Antity.js is still in development and we would be glad to get all the help you can provide. To contribute please read contributor.md for detailed installation guide.
| Purpose | Choice | Motivation |
|---|---|---|
| repository | Github | hosting for software development version control using Git |
| package manager | npm | default node.js package manager |
| language | TypeScript | static type checking along with the latest ECMAScript features |
| module bundler | Rollup | advanced module bundler for ES6 modules |
| unit testing | Jest | delightful testing with a focus on simplicity |