Delete page "Dive-Deeper"
parent
a5514ecde2
commit
9ea75d324b
144
Dive-Deeper.md
144
Dive-Deeper.md
@ -1,144 +0,0 @@
|
|||||||
_In this page, the deeper understand of decorator will be handled._
|
|
||||||
|
|
||||||
# How basic decorators works?
|
|
||||||
|
|
||||||
In `nestlogged` package, there are three types of decorator.
|
|
||||||
|
|
||||||
1. Class Decorator (`LoggedInjectable`, `LoggedController`)
|
|
||||||
2. Method Decorator (`LoggedFunction`, `LoggedRoute`, `ShouldScoped`)
|
|
||||||
3. Parameter Decorator (`LoggedParam`, `InjectLogger`, `ScopeKey`)
|
|
||||||
|
|
||||||
There is a [good documentation](https://www.typescriptlang.org/docs/handbook/decorators.html) for this in typescriptlang.org.
|
|
||||||
|
|
||||||
# How does the logging decorator actually works?
|
|
||||||
|
|
||||||
In this section, the internal logic of the logging decorators are explained.
|
|
||||||
|
|
||||||
The main logic of this package is placed inside of method decorators.
|
|
||||||
But before explaining about real logging decorators (method decorators), you have to understand the parameter decorator.
|
|
||||||
|
|
||||||
## Metadata Reflection
|
|
||||||
|
|
||||||
If you open the package.json file and see what is the dependency, there is a package `reflect-meatadata`.
|
|
||||||
It is used by nestjs, but also nestlogged uses it very actively.
|
|
||||||
All `LoggedParam`, `InjectLogger`, `ScopeKey` is powered by `reflect-metadata` package.
|
|
||||||
|
|
||||||
It provides **Metadata Reflection API**, which gives functionality to add additional metadatas to a class.
|
|
||||||
It is pretty useful for saving the metadata in a decorator, and then use it in another decorator.
|
|
||||||
|
|
||||||
## Parameter Decorator Implementation Explained
|
|
||||||
|
|
||||||
If you look at `src/reflected.ts` file, there is implementations for parameter decorators.
|
|
||||||
What these decorators have in common is they store metadatas using metadata reflection api.
|
|
||||||
|
|
||||||
### `InjectLogger`
|
|
||||||
|
|
||||||
Let's look at `InjectLogger` function.
|
|
||||||
|
|
||||||
It's pretty small and simple.
|
|
||||||
|
|
||||||
What it does is putting the information of parameter to it's target.
|
|
||||||
|
|
||||||
When we put the `InjectLogger` to the parameter, the _the prototype of the class for an instance member_ (`target: any`), _The name of the member_ (`propertyKey: string | symbol`), and _The ordinal index of the parameter in the function’s parameter list_ (`parameterIndex: number`) are given to the `InjectLogger` function as the argument.
|
|
||||||
|
|
||||||
And call `Reflect.defineMetadata` to save the `parameterIndex` as `ScopedLogger` injection index.
|
|
||||||
|
|
||||||
The key (`scopedLogger`) when used to define metadata is defined at the almost top of the file, right bottom to the interfaces, as the constant symbol.
|
|
||||||
|
|
||||||
It will be used in both `LoggedFunction` and `LoggedRoute` which will explained later section.
|
|
||||||
|
|
||||||
### `LoggedParam`
|
|
||||||
|
|
||||||
At the right bottom to the `InjectLogger`, there is a function definition for `LoggedParam`.
|
|
||||||
|
|
||||||
It takes parameter name (`name: string`), and options for include or exclude path in object (`options?: {includePath?: ..., excludePath?: ...}`).
|
|
||||||
|
|
||||||
It returns parameter decorator function, which has same parameters with InjectLogger.
|
|
||||||
|
|
||||||
Since the decorator function is **returned after call**, it should be called to be applied.
|
|
||||||
|
|
||||||
```ts
|
|
||||||
function a(
|
|
||||||
@InjectLogger logger: ScopedLogger,
|
|
||||||
@LoggedParam param: string // it's illegal, error
|
|
||||||
@LoggedParam('param') param: string // Now it returns decorator function to be applied.
|
|
||||||
) {}
|
|
||||||
```
|
|
||||||
|
|
||||||
`LoggedParam` decorator can be used multiple time in one function.
|
|
||||||
|
|
||||||
Because of that, first thing decorator does is getting the existing decorator data.
|
|
||||||
Right after that, push a new data, then redefine the metadata.
|
|
||||||
|
|
||||||
### `ScopeKey`
|
|
||||||
|
|
||||||
It is simillar to the `LoggedParam` decorator implementation.
|
|
||||||
|
|
||||||
The only different thing is, `ScopeKey` sorts its metadata array by priority.
|
|
||||||
|
|
||||||
### `ShouldScoped`
|
|
||||||
|
|
||||||
Although `ShouldScoped` is a method decorator, it only defined metadata like `InjectLogger`, and it's very simple.
|
|
||||||
It just set `forceScopeKey` to true in method metadata.
|
|
||||||
Getting the metadata value which is not defined is same as undefined, so the default value is falsy.
|
|
||||||
|
|
||||||
## Method Decorator Implementation Explained
|
|
||||||
|
|
||||||
Now, look at `src/logged.ts`.
|
|
||||||
|
|
||||||
This file contains `LoggedFunction`, `LoggedRoute`, `LoggedInjectable`, `LoggedController`.
|
|
||||||
|
|
||||||
Most basic implementation is `LoggedFunction`.
|
|
||||||
Let's look inside of that.
|
|
||||||
|
|
||||||
`LoggedFunction` is a method decorator itself, and has three parameters.
|
|
||||||
1. _the prototype of the class for an instance member_ (`_target: any`) (prototype of the class that owns the method)
|
|
||||||
2. _The name of the member_ (`key: string`)
|
|
||||||
3. _The Property Descriptor for the member_ (`descriptor: TypedPropertyDescriptor<(...args: F) => Promise<R>>`)
|
|
||||||
|
|
||||||
First of all, it calls `loggerInit` function with _target.
|
|
||||||
|
|
||||||
The `loggerInit` function is at the top of file, taking _target: any as the only parameter.
|
|
||||||
When `loggerInit` is called, if _target (prototype) has no logger, make new logger with their name, and define it as its property.
|
|
||||||
|
|
||||||
Then, after loggerInit finished, the LoggedFunction takes initialized _target.logger, and descriptor.value as constant fn.
|
|
||||||
|
|
||||||
descriptor.value is the function that LoggedFunction decorator will be applied.
|
|
||||||
It is ensured to be function. if it is not a function, logger will warn about this, and stop applying the decorator.
|
|
||||||
|
|
||||||
The next step is taking all the metadata sets out. It will be explained later, so just keep that in mind.
|
|
||||||
|
|
||||||
Next four constants are just getting data of `InjectLogger`, `LoggedParam`, `ScopeKey`, `ShouldScoped` decorators.
|
|
||||||
|
|
||||||
and then it creates overrideFunction constant with overrideBuild.
|
|
||||||
What overrideBuild makes is just a function that has same parameter and return type with original descriptor.value.
|
|
||||||
But the difference is - it takes all parameter data, original function, logger, etc.
|
|
||||||
|
|
||||||
It is really important to understand the inside of the overrideBuild function.
|
|
||||||
|
|
||||||
It makes new ScopedLogger, print parameter logs, and injects logger, log return response.
|
|
||||||
|
|
||||||
If it catches any error while running original function, it catches and logs with injected logger, and then throws it back.
|
|
||||||
|
|
||||||
It takes input, works, and return output same as original function, but internally logging things.
|
|
||||||
|
|
||||||
After building core function that will be override on top of original function, it overrides the function.
|
|
||||||
|
|
||||||
_target[key] overriding is for class decorators like `LoggedInjectable` or `LoggedController`,
|
|
||||||
and descriptor.value overriding is for direct decorators like `LoggedRoute` or `LoggedFunction`.
|
|
||||||
|
|
||||||
`LoggedRoute` and `LoggedFunction` has almost same implementation, but `LoggedRoute` takes `path` and `method` metadata that is defined by nestjs.
|
|
||||||
|
|
||||||
`@nestjs/common` internally uses reflect-metadata, and saves path and method as metadata.
|
|
||||||
|
|
||||||
So it gets the metadata from "path" and "method" key, and then use it inside of overrideBuild to build a log.
|
|
||||||
|
|
||||||
The logging decorator lastly does save all the metadata previously got.
|
|
||||||
|
|
||||||
This is required, because when it overrides function, the metadata which was originally included in function is wiped by new function.
|
|
||||||
|
|
||||||
This problem is really serious, because it wipes the path, method, guards, everything it has.
|
|
||||||
Without redefining step, the function will not behave like you expected.
|
|
||||||
NestJS will not detect the method as the request handler, or guard does not work, etc.
|
|
||||||
|
|
||||||
So, redefining every single original metadata is really important step.
|
|
Loading…
x
Reference in New Issue
Block a user