@ControllerComponent(
'/Users'
)
export class UserController implements Controller {
@Input() user: User;
@Input() password: string;
@Post()
postUser(@Res res: Response, @Req req: Request) {
console.log('Utente letto dal body =>', this.user);
console.log('password', this.password);
console.log('Send User by UserController');
res.status(200).send({'utente': this.user});
}
@Get()
getUser(@Res res: Response, @Req req: Request) {
console.log('Utente letto dal body =>', this.user);
console.log('Get User by UserController');
res.status(200).send({message: 'Get User OK'});
}
@Secured(['ADMIN'])
@Get('/admin')
getAdmin(@Res res: Response){
console.log('Get Admin by UserController');
res.status(200).send({message: 'Get Admin OK'});
}
}
npm install
npm run start:dev
Nella folder src
troverai un esempio di creazione di un Controller il cui scopo è quello di fornire i metodi di accesso ad una risorsa tramite una API definita dagli endpoint che vengono decorati.
Nota sui tipi: il progetto inizialmente ha la maggior parte del codice in any
e quindi debolmente tipizzata. Più avanti verranno create le interfaccie e i type necessari a fornire a TypeScript le dovute informazioni che faranno si che il service di TypeScript ci possa fornire feedback su errori nel codice.
Ad esempio, perchè utilizzare una interfaccia Controller? L'interfaccia ci permette di definire dei vincoli e avere quindi più certezza che il codice venga scritto in maniera più corretta.
Il controller è una classe che gestisce degli Endpoint, delle Richieste, delle Response, ma anche una serie di parametri provenienti dalla Query String o dal Body della Richiesta.
Ha dei metodi che associamo agli HTTP VERBS tramite i decoratori: @Get
, @Post
, @Put
o @Delete
, ma abbiamo anche altri decoratori come @Controller
, @Server
, @NodAngular
o @Input
che ci permettono di creare un codice molto simile ad Angular lato backend.
Si suddividono in meta dati per il: Server (metaServer), per il Controller (metaController), per le Routes del controller (metaRoutes).
## Creazione dell'Handler
for (const nomeMetodo of Object.keys(routes)) {
const routeHandler = () => {
const handler = controller[nomeMetodo].apply(controller);
return handler;
};
router[routes[nomeMetodo].method].apply(router, [routes[nomeMetodo].path, routeHandler])
}
Questo codice cicla per ogni metodo (decorato da un HTTP VERB) e costruisce l'handler dell'endpoint in base al metodo stesso, applicando gli argomenti che passiamo. Di default potrei pensare che i miei metodi ricevano req, res, next e avere una classe Controller con le seguenti proprietà:
private req!: Request;
private res!: Response;
private next!: NextFunction
questo mi porta ad avere nella gestione dell'handler la costruzione dei 3 parametri req, res e next:
controller.req = req;
controller.res = res;
const handler = controller[nomeMetodo].apply(controller);
Oppure pensare di dichiararli nella funzione stessa:
getUser(req: Request, res: Response, next: NextFunction) {
ma una funzione riceverà anche altri argomenti. Come gestirli? Prima di tutto scegliamo la soluzione che evidenzia per ogni metodo, quale argomento prendere tra req, res e next.
Dovrò costruire il mio handler così:
const handler = controller[nomeMetodo].apply(controller, [req, res, next]);
Ma quell'elenco di parametri che passo all'apply la devo costruire dinamicamente, prevedendo che un parametro potrei non passarlo o passarli in ordine diverso dal classico req, res e next.
A runtime ho una funzione che mi restituisce un array di argomenti, assegnando a ciasuno indice dell'array il corretto valore:
for (const { index, type } of params) {
console.log(index, type)
switch (type) {
case 'REQ': args[index] = req; break;
case 'RES': args[index] = res; break;
case 'NEXT': args[index] = next; break;
}
}
Supponiamo di aver passato solo un oggetto @Res alla funzione del controller. Lo snip di codice è un ciclo for che prende un elenco di parametri del metodo decorato tramite destructuring costruisco le variabili index e type. Grazie ad una switch assegno REQ, RES e NEXT all'index i-esimo dell'array. Per cui se RES sarà il primo parametro, allora l'array nella 0-esima posizione avrà proprio un oggetto res. Come costruisco il Decoratore dei Parametri?
export const paramDecoratorFactory = (type: any) => {
return function (target: any, methodName: string | symbol, index: number) {
target.metaParams = target.metaParams || { params: []};
if (target.metaParams.params[methodName] === undefined) {
target.metaParams.params[methodName] = [];
}
target.metaParams.params[methodName].push({ index, type });
};
}
export const Req = paramDecoratorFactory('REQ');
export const Res = paramDecoratorFactory('RES');
export const Next = paramDecoratorFactory('NEXT');
Per ogni metodo del controller decorato con dei @PARAMS, creo un array di oggetti in cui inserisco i seguenti dati: {index. type}
, cioè la posizione del parametro ed il tipo.
Dobbiamo prevedere l'acquisizione di dati dal body di una richiesta. Il primo passaggio è inserire la libreria body-parser
con le type definition @types/body-parser
.
export const Body = paramDecoratorFactory('BODY');
case 'BODY':
args[index] = getParam(req, 'body');
break;
const getParam = (source: any, paramType: string | null): any => {
return paramType ? source[paramType] : source;
}
L'idea però è quella di avere nel controller la possibilità non solo di indicare la volontà di prelevare il body e salvarlo in una variabile, ma anche di prendere una parte del body stesso:
postUser(@Res res: Response, @Req req: Request, @Body('user') dati: any) {
Per cui dobbiamo prevedere di costruire un array che mi indica anche la porzione di body di interesse: target.metaParams.params[methodName].push({ index, type, name });
Per questo aggiungiamo name a: {index, type, name}
.
ed il decoratore sarà:
const bodyDecoratorFactory = (type: string) => {
return (name?: string) => {
return (target: any, method: string | symbol, index: number) => {
console.log('BODYNAME', name);
target.metaParams = target.metaParams || { params: []};
if (target.metaParams.params[method] === undefined) {
target.metaParams.params[method] = [];
}
target.metaParams.params[method].push({ index, type, name });
}
}
}
Cioè una decorator factory che restituisce una funzione che acquisisce il nome della variabile che cerchiamo nel body ed infine il decorator vero e proprio. Insomma High Order Function a tutta birra!
Possiamo notare nel codice che bodyDecoratorFactory
è proprio simile a paramDecoratorFactory
a meno di una funzione di ritorno. Per ora le teniamo distinte.