From 2badacbbede4e2892c5cd383872a27a3ad01ed1e Mon Sep 17 00:00:00 2001 From: Shinwoo PARK Date: Sun, 10 Dec 2023 21:22:19 +0900 Subject: [PATCH] Updated Tutorial (markdown) --- Tutorial.md | 239 +++++++++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 238 insertions(+), 1 deletion(-) diff --git a/Tutorial.md b/Tutorial.md index 247e38a..68a8958 100644 --- a/Tutorial.md +++ b/Tutorial.md @@ -620,4 +620,241 @@ 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. \ No newline at end of file +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. \ No newline at end of file