From 66a92fc728d55aa8252766bf6466db3cc59a7f04 Mon Sep 17 00:00:00 2001 From: p-sw Date: Wed, 26 Mar 2025 22:29:13 +0000 Subject: [PATCH] Delete page "Tutorial" --- Tutorial.md | 854 ---------------------------------------------------- 1 file changed, 854 deletions(-) delete mode 100644 Tutorial.md diff --git a/Tutorial.md b/Tutorial.md deleted file mode 100644 index e3ec3a8..0000000 --- a/Tutorial.md +++ /dev/null @@ -1,854 +0,0 @@ -# Getting Started - -Let's start from making a normal NestJS app. - -```ts -import { Controller, Get } from '@nestjs/common'; -import { AppService } from './app.service'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @Get() - getHello(): string { - return this.appService.getHello(); - } -} -``` - -**Since version 2.1.0, all decorator can be used with sync function.** - -```ts -import { Controller, Get } from '@nestjs/common'; -import { AppService } from './app.service'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @Get() - async getHello(): Promise { - return await this.appService.getHello(); - } -} -``` - -This is the template you will get when you start with `nest new [name]`. - -Now install `nestlogged` package. - -```sh -npm install nestlogged -``` - -Or, - -```sh -yarn add nestlogged -``` - -# Basic Usage -## Add Logs To Your Route - -In this package, logging decorator is separated by two kind. - -1. LoggedRoute is used for methods in Controllers, with method decorators like @Get, @Post, etc. - -2. LoggedFunction is used for methods in any other class methods, like service. - -So, we can add a LoggedRoute decorator in controller example above. - -```ts -import { Controller, Get } from '@nestjs/common'; -import { AppService } from './app.service'; -import { LoggedRoute } from 'nestlogged'; // Import... - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() // Add! - @Get() - async getHello(): Promise { - return await this.appService.getHello(); - } -} -``` - -It will automatically add logs to your route, and print it when the route method is called. - -```md -[Nest] 2378244 - 12/10/2023, 6:53:49 PM LOG [AppController] HIT HTTP AppController::/[GET] (getHello) -[Nest] 2378244 - 12/10/2023, 6:53:49 PM LOG [AppController] RETURNED RESPONSE AppController::/[GET] (getHello) -``` - -Note that it uses Logger class provided by `@nestjs/common` package. - -### Changing the logging path - -Basically LoggedRoute decorator automatically detect its path and http method, and build log with that. - -But if you want, You can also provide its path to LoggedRoute decorator. - -```ts -import { Controller, Get } from '@nestjs/common'; -import { AppService } from './app.service'; -import { LoggedRoute } from 'nestlogged'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute('helloWorld') // give a path you want to log - @Get() - async getHello(): Promise { - return await this.appService.getHello(); - } -} -``` - -In this way, the LoggedRoute decorator will log like this: - -```md -[Nest] 2442776 - 12/10/2023, 7:04:49 PM LOG [AppController] HIT HTTP AppController::helloWorld[GET] (getHello) - ^^^^^^^^^^ -[Nest] 2442776 - 12/10/2023, 7:04:49 PM LOG [AppController] RETURNED RESPONSE AppController::helloWorld[GET] (getHello) - ^^^^^^^^^^ - Here, just path / changed to 'helloWorld' that we provided to decorator! -``` - -## Add Logs To Your Class Methods - -You can also add logs to your class methods, as simple as the LoggedRoute decorator above. - -For example, in `app.service.ts` file: - -```ts -import { Injectable } from '@nestjs/common'; - -@Injectable() -export class AppService { - async getHello(): Promise { - return 'Hello World!'; - } -} -``` - -Now, add `LoggedFunction` decorator: - -```ts -import { Injectable } from '@nestjs/common'; -import { LoggedFunction } from 'nestlogged'; // Import... - -@Injectable() -export class AppService { - @LoggedFunction // Add! - async getHello(): Promise { - return 'Hello World!'; - } -} -``` - -The `LoggedFunction` decorator will automatically add logs to the call time. - -When the `getHello` function called, it will automatically prints: - -```md -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppService] CALL getHello -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppService] RETURNED getHello -``` - -Since it is called from the http route `getHello`, the full log will printed like below. - -```md -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppController] HIT HTTP AppController::helloWorld[GET] (getHello) -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppService] CALL getHello -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppService] RETURNED getHello -[Nest] 2459083 - 12/10/2023, 7:08:11 PM LOG [AppController] RETURNED RESPONSE AppController::helloWorld[GET] (getHello) -``` - -In this log, you can see that the `getHello` in `AppController` is calling the `getHello` function in `AppService`, and then return, and return. - -You can easily see the flow of the http request. Pretty cool, isn't it? - -## Parameter Logging - -You can also add a parameter log when the function or route method is called. - -Let's modify the previous app and add some parameter, so we can get some arguments. - -`app.controller.ts` -```ts -import { Controller, Get, Query } from '@nestjs/common'; -import { AppService } from './app.service'; -import { LoggedRoute } from 'nestlogged'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() - @Get() - async getHello( - @Query() { name = 'World' }: { name?: string }, - ): Promise { - return await this.appService.getHello(name); - } -} - -``` - -`app.service.ts` -```ts -import { Injectable } from '@nestjs/common'; -import { LoggedFunction } from 'nestlogged'; - -@Injectable() -export class AppService { - @LoggedFunction - async getHello(name: string): Promise { - return `Hello ${name}!`; - } -} -``` - -It's not that much different. - -We add a query parameter `name`, pass to `AppService`'s `getHello`, return `Hello [name]`. - -Now, if we add a query `?name=NestLogged` to url, you can get "Hello NestLogged!" instead of "Hello World!". - -But still, decorators does not add any log for parameter. - -What if we want to track the value of parameter `name`? - -Just add @LoggedParam to parameter. - -`app.controller.ts` -```ts -import { Controller, Get, Query } from '@nestjs/common'; -import { AppService } from './app.service'; -import { LoggedParam, LoggedRoute } from 'nestlogged'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() - @Get() - async getHello( -// vvvvvvvvvvvvvvvvvvvvv - @LoggedParam('query') @Query() { name = 'World' }: { name?: string }, - ): Promise { - return await this.appService.getHello(name); - } -} -``` - -`app.service.ts` -```ts -import { Injectable } from '@nestjs/common'; -import { LoggedFunction, LoggedParam } from 'nestlogged'; - -@Injectable() -export class AppService { - @LoggedFunction - async getHello(@LoggedParam('name') name: string): Promise { -// ^^^^^^^^^^^^^^^^^^^^ - return `Hello ${name}!`; - } -} -``` - -The `LoggedParam` decorator will make a log for your parameter. - -After change, the log will change like below: - -```md - vvvvvvvvvvvvvvvvvvvvvvvvvvvvvvvv -[Nest] 2555365 - 12/10/2023, 7:28:22 PM LOG [AppController] HIT HTTP AppController::/[GET] (getHello) WITH query={"name":"NestLogged"} - vvvvvvvvvvvvvvvvvvvv -[Nest] 2555365 - 12/10/2023, 7:28:22 PM LOG [AppService] CALL getHello WITH name=NestLogged -[Nest] 2555365 - 12/10/2023, 7:28:22 PM LOG [AppService] RETURNED getHello -[Nest] 2555365 - 12/10/2023, 7:28:22 PM LOG [AppController] RETURNED RESPONSE AppController::/[GET] (getHello) -``` - -The first parameter of LoggedParam decides its name. - -The second parameter, `options`, will be explained right after. - -### Object Logging - -So, we logged parameter. (Yay) - -Now we're gonna make a simple user registration. - -We'll make a route and service. - -`types.d.ts` -```ts -export interface RegistrationBody { - username: string; - password: string; - email: string; -} -``` - -`app.controller.ts` -```ts -import { BadRequestException, Body, Controller, Post } from '@nestjs/common'; -import { AppService } from './app.service'; -import { LoggedParam, LoggedRoute } from 'nestlogged'; -import { RegistrationBody } from './types'; - -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() - @Post('register') - async userRegistration( - @LoggedParam('data') @Body() data: RegistrationBody, - ): Promise { - if ( - (await this.appService.checkUsernameDuplication(data.username)) || - (await this.appService.checkEmailDuplication(data.email)) - ) { - throw new BadRequestException('Duplicated User'); - } - - await this.appService.addUser(data); - } -} -``` - -`app.service.ts` -```ts -import { RegistrationBody } from './types.d'; -import { Injectable } from '@nestjs/common'; -import { LoggedFunction, LoggedParam } from 'nestlogged'; - -@Injectable() -export class AppService { - @LoggedFunction - async checkUsernameDuplication( - @LoggedParam('username') username: string, - ): Promise { - /* Magic Function */ - } - - @LoggedFunction - async checkEmailDuplication( - @LoggedParam('email') email: string, - ): Promise { - /* Magic Function */ - } - - @LoggedFunction - async addUser( - @LoggedParam('data') data: RegistrationBody, - ) { - /* Magic Function */ - } -} -``` - -For conciseness, we skipped service function implementations. - -For now, let's assume that all service functions works without problem, and check duplication methods are always returning false. - -Now, we're going to put the data using this route. - -```sh -$ curl -X POST "http://localhost:3000/register" -d "{\"username\": \"NestLogged\", \"password\": \"SuperSecretForUs\", \"email\": \"nestlogged@worplo.com\"}" -H "Content-Type: application/json" -``` - -```md -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppController] HIT HTTP AppController::register[POST] (userRegistration) WITH data={"username":"NestLogged","password":"SuperSecretForUs","email":"nestlogged@worplo.com"} -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] CALL checkUsernameDuplication WITH username=NestLogged -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] RETURNED checkUsernameDuplication -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] CALL checkEmailDuplication WITH email=nestlogged@worplo.com -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] RETURNED checkEmailDuplication -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] CALL addUser WITH data={"username":"NestLogged","password":"SuperSecretForUs","email":"nestlogged@worplo.com"} -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppService] RETURNED addUser -[Nest] 2649573 - 12/10/2023, 7:50:09 PM LOG [AppController] RETURNED RESPONSE AppController::register[POST] (userRegistration) -``` - -Well, that works, but... Our super secret is revealed in log. - -That is a big problem. Fortunately, we have a solution for this. - -the `LoggedParam` decorator has a second parameter, `options`, for this situation. - -There is a two options, `includePath` and `excludePath`. - -We want to **exclude** our secret in the log, so we can use option `excludePath`. - -`app.controller.ts` -```ts -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() - @Post('register') - async userRegistration( -// vvvvvvvvvvvvvvvvvvvvvvvvvvvvvv - @LoggedParam('data', { excludePath: ['password'] }) - @Body() - data: RegistrationBody, - ): Promise { - if ( - (await this.appService.checkUsernameDuplication(data.username)) || - (await this.appService.checkEmailDuplication(data.email)) - ) { - throw new BadRequestException('Duplicated User'); - } - - await this.appService.addUser(data); - } -} -``` - -`app.service.ts` -```ts -import { RegistrationBody } from './types.d'; -import { Injectable } from '@nestjs/common'; -import { LoggedFunction, LoggedParam } from 'nestlogged'; - -@Injectable() -export class AppService { -// ... - @LoggedFunction - async addUser( -// vvvvvvvvvvvvvvvvvvvvvvvvvvvvvv - @LoggedParam('data', { excludePath: ['password'] }) data: RegistrationBody, - ): Promise { - return data; - } -} -``` - -Now, if we see the log: - -```md -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppController] HIT HTTP AppController::register[POST] (userRegistration) WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] CALL checkUsernameDuplication WITH username=NestLogged -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] RETURNED checkUsernameDuplication -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] CALL checkEmailDuplication WITH email=nestlogged@worplo.com -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] RETURNED checkEmailDuplication -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] CALL addUser WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppService] RETURNED addUser -[Nest] 2688311 - 12/10/2023, 7:57:17 PM LOG [AppController] RETURNED RESPONSE AppController::register[POST] (userRegistration) -``` - -There is no secret anymore. Super simple, right? - -We can also use `includePath` like: `@LoggedParam('data', { includePath: ['username', 'email'] })`. - -# Advanced Usage -## Logger Injection - -So, that was basic usage. - -But how can we get the logger, print whatever logs we want, while not losing the advantage of this decorator? - -You can easily inject the logger using `InjectLogger` decorator. - -Like this: - -```ts -@Controller() -export class AppController { - constructor(private readonly appService: AppService) {} - - @LoggedRoute() - @Post('register') - async userRegistration( - @LoggedParam('data', { excludePath: ['password'] }) - @Body() - data: RegistrationBody, - @InjectLogger logger: ScopedLogger, - ): Promise { - if ( - (await this.appService.checkUsernameDuplication(data.username)) || - (await this.appService.checkEmailDuplication(data.email)) - ) { - logger.error('Duplicated data!!'); - throw new BadRequestException('Duplicated User'); - } - - logger.log('Now calling addUser..'); - await this.appService.addUser(data); - } -} -``` - -In later chapter, we will shortly explain what is `ScopedLogger`, and how can we really use it. - -But, for now, just focus on logger injection. - -Because we added `@InjectLogger`, the LoggedRoute knows where to put `ScopedLogger`. - -So it can create and inject the new logger when the function is called. - -You can also inject new `ScopedLogger` to any other methods. - -For example, in service: - -`app.service.ts` -```ts -// ... - @LoggedFunction - async addUser( - @LoggedParam('data', { excludePath: ['password'] }) data: RegistrationBody, - @InjectLogger logger?: ScopedLogger - ): Promise { - logger?.log('Now creating a new user...'); - return data; - } -// ... -``` - -> Note: even if logger parameter is not provided when `addUser` method is called, - -Like before, we can call service normally in the controller. We don't have to fix anything. - -```md -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppController] -> userRegistration: HIT HTTP AppController::register[POST] (userRegistration) WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] CALL checkUsernameDuplication WITH username=NestLogged -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] RETURNED checkUsernameDuplication -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] CALL checkEmailDuplication WITH email=nestlogged@worplo.com -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] RETURNED checkEmailDuplication -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppController] -> userRegistration: Now calling addUser.. -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] CALL addUser WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppService] RETURNED addUser -[Nest] 2771625 - 12/10/2023, 8:18:05 PM LOG [AppController] -> userRegistration: RETURNED RESPONSE AppController::register[POST] (userRegistration) -``` - -Except our call of injected logger, there is something slightly different. - -Now, it is time to the key of this package. **Scoped Logging**. - -## Scoped Logging - -Simple. - -When the user requested to our backend, there is a **flow**. - -if username is existing in our database, -``` -HIT /register -> CALL checkUsernameDuplication() -> RETURN true -> RETURN RESPONSE Forbidden -``` - -if email is existing in our database, -``` -HIT /register -> CALL checkUsernameDuplication() -> RETURN false -> CALL checkEmailDuplication() -> RETURN true -> RETURN RESPONSE Forbidden -``` - -if successful, -``` -HIT /register -> CALL checkUsernameDuplication() -> RETURN false -> CALL checkEmailDuplication() -> RETURN false -> CALL addUser() -> RETURN RegistrationBody -> RETURN RESPONSE Ok -``` - -It is single line. - -Then, how about reading its flow? - -If there is a lot of request, it's hard to read a single flow. - -So, ScopedLogger, makes a log **scoped**. - -It's really simple to apply. - -Just put existing ScopedLogger to the injected logger parameter place. - -`app.controller.ts` -```ts -// ... - @LoggedRoute() - @Post('register') - async userRegistration( - @LoggedParam('data', { excludePath: ['password'] }) - @Body() - data: RegistrationBody, - @InjectLogger logger: ScopedLogger, - ): Promise { - if ( - (await this.appService.checkUsernameDuplication(data.username, logger)) || // added logger - (await this.appService.checkEmailDuplication(data.email, logger)) // added logger - ) { - logger.error('Duplicated data!!'); - throw new BadRequestException('Duplicated User'); - } - - logger.log('Now calling addUser..'); - await this.appService.addUser(data, logger); // added logger - } -// ... -``` - -In this way, logger will be like this: - -```md -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: HIT HTTP AppController::register[POST] (userRegistration) WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkUsernameDuplication: CALL checkUsernameDuplication WITH username=NestLogged -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkUsernameDuplication: There is no duplication! -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkUsernameDuplication: RETURNED checkUsernameDuplication -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkEmailDuplication: CALL checkEmailDuplication WITH email=nestlogged@worplo.com -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkEmailDuplication: There is no duplication! -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> checkEmailDuplication: RETURNED checkEmailDuplication -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: Now calling addUser.. -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> addUser: CALL addUser WITH data={"username":"NestLogged","email":"nestlogged@worplo.com"} -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> addUser: Now creating a new user... -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: -> addUser: RETURNED addUser -[Nest] 2859432 - 12/10/2023, 8:34:57 PM LOG [AppController] -> userRegistration: RETURNED RESPONSE AppController::register[POST] (userRegistration) -``` - -You see the flow? - -It's little longer than before, but you can easily see the flow. - -But it's still hard to see the **single** flow. - -For instance, if user 'a' and 'b' simultaneously call `/register` endpoint, it will be confusing to read only user `a`'s log. - -### Adding a Scope Key - -We made additional solution for this, recently. -The **Scoped Key**. - -You can add scoped key using `ScopeKey` decorator. - -```ts -// ... - @LoggedRoute() - @Post('register') - async userRegistration( - @LoggedParam('data', { excludePath: ['password'] }) - @ScopeKey('username', { path: ['username'] }) // add this! - @Body() - data: RegistrationBody, - @InjectLogger logger: ScopedLogger, - ): Promise { - if ( - (await this.appService.checkUsernameDuplication(data.username, logger)) || - (await this.appService.checkEmailDuplication(data.email, logger)) - ) { - logger.error('Duplicated data!!'); - throw new BadRequestException('Duplicated User'); - } - - logger.log('Now calling addUser..'); - await this.appService.addUser(data, logger); - } -// ... -``` - -```sh -$ curl -X POST http://localhost:3000/register -d "{\"username\": \"A\", \"password\": \"SuperSecret\", \"email\": \"NestLogged@worplo.com\"}" -H "Content-Type: application/json" -``` - -```md -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): HIT HTTP AppController::register[POST] (userRegistration) WITH data={"username":"A","email":"NestLogged@worplo.com"} -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkUsernameDuplication: CALL checkUsernameDuplication WITH username=A -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkUsernameDuplication: There is no duplication! -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkUsernameDuplication: RETURNED checkUsernameDuplication -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkEmailDuplication: CALL checkEmailDuplication WITH email=NestLogged@worplo.com -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkEmailDuplication: There is no duplication! -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> checkEmailDuplication: RETURNED checkEmailDuplication -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): Now calling addUser.. -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> addUser: CALL addUser WITH data={"username":"A","email":"NestLogged@worplo.com"} -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> addUser: Now creating a new user... -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): -> addUser: RETURNED addUser -[Nest] 2950529 - 12/10/2023, 8:51:46 PM LOG [AppController] -> userRegistration(username=A): RETURNED RESPONSE AppController::register[POST] (userRegistration) -``` - -Now, even if user `A` and `B` calls backend at the same time, we can recognize what is user `A`'s log! - -### Scope Key Advanced Technique - -We made `ScopeKey` decorator dynamically, to useful in almost every situation. - -For example, it easily handles OR. - -Let's pick a method from our test cases. - -```ts -// ... - @LoggedFunction - async testOrScopedLogging( - @LoggedParam("key") - @ScopeKey("scopekey-a", { path: "a" }) - @ScopeKey("scopekey-b", { path: "b" }) - key: { a: string } | { b: string }, - @InjectLogger logger?: ScopedLogger - ) { - logger.log(JSON.stringify(key)); - } -// ... - -tester.testOrScopedLogging({ a: "asdf" }); -tester.testOrScopedLogging({ b: "qwer" }); -``` - -Log: - -```md -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-a=asdf): CALL testOrScopedLogging WITH key={"a":"asdf"} -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-a=asdf): {"a":"asdf"} -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-b=qwer): CALL testOrScopedLogging WITH key={"b":"qwer"} -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-b=qwer): {"b":"qwer"} -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-a=asdf): RETURNED testOrScopedLogging -[Nest] 2999438 - 12/10/2023, 9:00:27 PM LOG [LoggedMethodsClass] -> testOrScopedLogging(scopekey-b=qwer): RETURNED testOrScopedLogging -``` - -If there is multiple ScopeKey decorator, it will try every single time, and only filter successful ones, and then select one by priority. - -You can give a priority in option. - -There is another test case that is simply illustrate the priority system. - -```ts -// ... - @LoggedFunction - async testPriorityScopedLogging( - @LoggedParam("key") - @ScopeKey("scopekey-a", { path: "a", priority: 0.5 }) - @ScopeKey("scopekey-b", { path: "b" }) // default 1 - key: { a?: string; b?: string }, - // if both a and b are undefined, set scope to nothing - @InjectLogger logger?: ScopedLogger - ) { - logger.log(JSON.stringify(key)); - } -// ... - -tester.testPriorityScopedLogging({ a: "asdf", b: "qwer" }); -tester.testPriorityScopedLogging({ a: "asdf" }); -tester.testPriorityScopedLogging({ b: "qwer" }); -tester.testPriorityScopedLogging({}); -``` - -```md -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): CALL testPriorityScopedLogging WITH key={"a":"asdf","b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): {"a":"asdf","b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): CALL testPriorityScopedLogging WITH key={"a":"asdf"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): {"a":"asdf"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): CALL testPriorityScopedLogging WITH key={"b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): {"b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: CALL testPriorityScopedLogging WITH key={} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: {} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): RETURNED testPriorityScopedLogging -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): RETURNED testPriorityScopedLogging -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): RETURNED testPriorityScopedLogging -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: RETURNED testPriorityScopedLogging -``` - -This is called asynchronously, so it's little massive, but still can find the flow. - -Let's sort it. - -`First call with {"a": "asdf", "b": "qwer"}` -```md -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): CALL testPriorityScopedLogging WITH key={"a":"asdf","b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): {"a":"asdf","b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): RETURNED testPriorityScopedLogging -``` - -`Second call with {"a": "asdf"}` -```md -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): CALL testPriorityScopedLogging WITH key={"a":"asdf"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): {"a":"asdf"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-a=asdf): RETURNED testPriorityScopedLogging -``` - -`Third call with {"b": "qwer"}` -```md -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): CALL testPriorityScopedLogging WITH key={"b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): {"b":"qwer"} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging(scopekey-b=qwer): RETURNED testPriorityScopedLogging -``` - -`Fourth call with {}` -```md -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: CALL testPriorityScopedLogging WITH key={} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: {} -[Nest] 3014294 - 12/10/2023, 9:03:37 PM LOG [LoggedMethodsClass] -> testPriorityScopedLogging: RETURNED testPriorityScopedLogging -``` - -See the function definition first. -There is a two ScopeKey decorator. first is named "scopekey-a", path is "a", and priority is 0.5. -Second one is named "scopekey-b", path is "b", and priority is not provided but default is 1. - -So, in the first call, both of them are existed but the scopekey-b was chosen, because the priority was bigger than scopekey-a. - -But there is no value in scopekey-b path, it will choose scopekey-a as the scope key. - -If there is nothing, no scope key will be chosen, no error occurred. - -When you need some debugging (warning when there is no scope key chosen), you can use `ShouldScoped` function decorator. - -```ts -// ... - @LoggedFunction - @ShouldScoped // Warn if there is no valid scopekey - async testShouldScopedLogging( - @LoggedParam("key") - @ScopeKey("scopekey") - key?: string, - @InjectLogger logger?: ScopedLogger - ) { - logger.log(key); - } -// ... -tester.testShouldScopedLogging("asdf"); -tester.testShouldScopedLogging(); -``` - -```md -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging(scopekey=asdf): CALL testShouldScopedLogging WITH key=asdf -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging(scopekey=asdf): asdf -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging(scopekey=asdf): RETURNED testShouldScopedLogging - - -[Nest] 3069515 - 12/10/2023, 9:15:14 PM WARN [LoggedMethodsClass] -> testShouldScopedLogging: ScopeKey in ShouldScope cannot be falsy value (undefined) -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging: CALL testShouldScopedLogging WITH key=undefined -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging: undefined -[Nest] 3069515 - 12/10/2023, 9:15:14 PM LOG [LoggedMethodsClass] -> testShouldScopedLogging: RETURNED testShouldScopedLogging -``` - -As you can see, it's warning you about falsy value of scope key, but it's not error. -It will not block you for falsy value, or not existing value. -If it is, then that's a bug. (please open a issue) - -## Logged Classes - -If it is hard to add a `LoggedFunction` or `LoggedRoute` to every single method, here is another solution. - -You can use `LoggedInjectable` or `LoggedController` to replace existing `Injectable` or `Controller` decorator from `@nestjs/common`. - -It simply apply `LoggedFunction` or `LoggedRoute` to every single method, and apply `Injectable` or `Controller` decorator to the target class. - -Use like this: - -```ts -@LoggedInjectable() -class LoggedClass { - async testParameterLoggingWithoutInjection(@LoggedParam("key") key: number) { - console.log(key); - } -// ... -} - -tester = new LoggedClass(); - -tester.testParameterLoggingWithoutInjection(1); -``` - -It is same as applying `LoggedFunction` to the method. - -But still, if you don't want to apply logging decorator to some methods, you have to apply logging decorator to other methods one by one. - -## Return Value Logging \ No newline at end of file