This small library is TS Metadata Organizer based on reflect-metadata
. "MA-TE" is for "ME-TA" in reverse order.
It allows you to define, manipulate, and read metadata at different levels, including classes, properties, methods, and parameters.
The primary purpose of Mate is to streamline the use of metadata, making it simpler and more efficient to handle.
You can install Mate from the npm registry.
npm install @prostojs/mate
After installation, you can start using Mate to define and handle metadata in your TypeScript project. Below is an example that showcases different levels of metadata usage.
import { TMateParamMeta } from '@prostojs/mate';
import { Mate } from '@prostojs/mate';
// Define your metadata interfaces
interface TClassMeta {
name: string;
model: string;
fromParam: string;
manufacturer: string;
data: string[];
inherit?: bolean;
}
interface TPropMeta<TParamMeta> {
description: string;
params: (TMateParamMeta & TParamMeta)[];
inherit?: bolean
}
interface TParamMeta {
required: boolean;
}
const mate = new Mate<TClassMeta, TPropMeta<TParamMeta>>('robot-meta', {
readReturnType: true,
readType: true,
collectPropKeys: true,
inherit(classMeta, targetMeta, level, prop) {
return !!targetMeta?.inherit || !!classMeta?.inherit;
},
});
// Annotate your class
@mate.decorate('name', 'NXT')
@mate.decorate('model', 'EV3')
@mate.decorate('manufacturer', 'LEGO')
@mate.decorate('data', 'NXT-0001', true)
class Robot {
@mate.decorate('description', 'Robotic arm for grabbing objects')
arm: string;
@mate.decorate('description', 'Sensors to detect objects')
sensors(
@mate.decorateClass('fromParam', 'hoised value')
@mate.decorate('required', true)
input: string
): string {
return 'Activated sensors with input: ' + input;
}
}
// Read the metadata
let classMeta = mate.read(Robot);
console.log(classMeta);
// Expected Output:
// {
// name: 'NXT',
// model: 'EV3',
// fromParam: 'hoised value',
// manufacturer: 'LEGO',
// data: ['NXT-0001'], // Array type metadata
// properties: ['arm', 'sensors'],
// type: Function, // Type of Robot
// returnType: undefined // Only for methods
// }
// read the property metadata
let armMeta = mate.read(Robot, 'arm');
console.log(armMeta);
// Expected Output:
// {
// description: 'Robotic arm for grabbing objects',
// params: [],
// type: Function, // Type of 'arm'
// returnType: undefined // Only for methods
// }
// read the method metadata
let sensorsMeta = mate.read(Robot, 'sensors');
console.log(sensorsMeta);
// Expected Output:
// {
// description: 'Sensors to detect objects',
// params: [{ type: String, required: true }],
// type: Function, // Type of 'sensors'
// returnType: String // Return type of 'sensors'
// }
It's possible to enable metadata inheritance and set some rules on whether it should be inherited from parent classes or not
Simply use true
on inherit
key to enable metadata inheritance
const mate = new Mate<MyMeta>('my-metadata-key', {
inherit: true, // this will enable metadata inheritance for all the classes
})
This example illustrates how to set a callback with logic to decide whether to inherit or not
const mate = new Mate<MyMeta>('my-metadata-key', {
inherit(classMeta, prop, methodMeta) {
if (prop) {
// if it is a method metadata
// we require a key `inherit` to be true for inheritance
return !!methodMeta?.inherit || !!(classMeta?.inherit && !methodMeta)
}
// otherwise (in case of class metadata) we require a key `inherit`
// to be true for inheritance on class metadata
return !!classMeta?.inherit
}
})
-
Mate<TClassMeta, TPropMeta>
: The main Mate class. It provides methods to decorate and read metadata. -
decorate(key, value?, isArray?, level?)
: A method for adding metadata to classes, properties, methods, and parameters. -
decorateClass(key, value?, isArray?)
: A method for hoisting metadata from methods or parameters to the class level. -
read(target, prop?, paramIndex?)
: A method for reading metadata.
It is recomended to include reflect-metadata
dependency to rely on the original implementation. Although it's still possible to use @prostojs/mate
with no dependency on reflect-metadata
because it already includes its own limited implementation of reflect-metadata which ships only the features used by @prostojs/mate
such as:
-
getOwnMetadata(key, target, prop?)
-
defineMetadata(key, data, target, prop?)
-
metadata(key, data)