Refactor to default to structured logging.
- Added `flattenMessage` and `FlattenedLogMessage` to default to structured logging. - Rework the logic formatting messages for the ConsoleLogger. - Add a more comprehensive README.
This commit is contained in:
@ -1,16 +1,52 @@
|
||||
import { LogMessage, LogLevel } from './log-message';
|
||||
import LogAppender from './log-appender';
|
||||
import { LogLevel, LogMessage } from './log-message';
|
||||
import { LogAppender } from './log-appender';
|
||||
|
||||
export type DeferredMsg = () => string | object;
|
||||
export type MessageType = string | DeferredMsg | object;
|
||||
export type DeferredMsg = () => string | Record<string, unknown>;
|
||||
export type MessageType = string | DeferredMsg | Record<string, unknown>;
|
||||
|
||||
export function isDeferredMsg(msg: MessageType): msg is DeferredMsg {
|
||||
return typeof msg === 'function';
|
||||
}
|
||||
|
||||
/**
|
||||
* Logger class for logging messages.
|
||||
*
|
||||
* Loggers are used to log messages at different levels. The levels are, in
|
||||
* order of increasing severity: TRACE, DEBUG, LOG, INFO, WARN, ERROR, FATAL.
|
||||
* Log messages can be logged at a specific level by calling the corresponding
|
||||
* method on the logger instance. For example, to log a message at the INFO
|
||||
* level, call the *info* method on the logger.
|
||||
*
|
||||
* Loggers have a threshold level, which is the minimum level of log message
|
||||
* that will be logged by the logger. Log messages with a level below the
|
||||
* threshold will be ignored. The threshold level can be set when creating a
|
||||
* logger, or by calling the *setThreshold* method on an existing logger. If a
|
||||
* threshold is not set, the logger will use the threshold of its parent
|
||||
* logger.
|
||||
*
|
||||
* Loggers are hierarchical, with the hierarchy defined by the logger name.
|
||||
* The heirarchy is tracked by the children: child loggers have a link to their
|
||||
* parents, while parent loggers do not have a list of their children.
|
||||
*
|
||||
* When a log message is logged by a logger, it is sent to all of the appenders
|
||||
* attached to the logger and to the parent logger. Appenders can be attached
|
||||
* to any Logger instance, but, because of this upwards propogation, appenders
|
||||
* are usually attached to the root logger, which is an ancestor of all loggers
|
||||
* created by the same LogService instance.
|
||||
*
|
||||
* Loggers are typically created and managed by a LogService instance. The
|
||||
* *LogService#getLogger* method is used to get a logger by name, creating it
|
||||
* if necessary. When creating a new logger, the parent logger is determined by
|
||||
* the longest existing logger name that is a prefix of the new logger name.
|
||||
* For more details, see *LogService#getLogger*.
|
||||
*/
|
||||
export class Logger {
|
||||
public appenders: LogAppender[] = [];
|
||||
|
||||
public constructor(
|
||||
public readonly name: string,
|
||||
private parentLogger?: Logger,
|
||||
public threshold?: LogLevel
|
||||
public threshold?: LogLevel,
|
||||
) {}
|
||||
|
||||
public createChildLogger(name: string, threshold?: LogLevel): Logger {
|
||||
@ -20,7 +56,7 @@ export class Logger {
|
||||
public doLog(
|
||||
level: LogLevel,
|
||||
message: Error | MessageType,
|
||||
stacktrace?: string
|
||||
stacktrace?: string,
|
||||
): void {
|
||||
if (level < this.getEffectiveThreshold()) {
|
||||
return;
|
||||
@ -31,20 +67,20 @@ export class Logger {
|
||||
level,
|
||||
message: '',
|
||||
stacktrace: '',
|
||||
timestamp: new Date()
|
||||
timestamp: new Date(),
|
||||
};
|
||||
|
||||
if (message === undefined || message === null) {
|
||||
logMsg.message = message;
|
||||
logMsg.stacktrace = stacktrace == null ? '' : stacktrace;
|
||||
} else if ((message as DeferredMsg).call !== undefined) {
|
||||
logMsg.message = (message as DeferredMsg)();
|
||||
logMsg.stacktrace = stacktrace == null ? '' : stacktrace;
|
||||
logMsg.stacktrace = stacktrace ?? '';
|
||||
} else if (message instanceof Error) {
|
||||
const error = message as Error;
|
||||
logMsg.error = error;
|
||||
logMsg.message = `${error.name}: ${error.message}`;
|
||||
logMsg.stacktrace = error.stack == null ? '' : error.stack;
|
||||
logMsg.stacktrace = stacktrace ?? error.stack ?? '';
|
||||
} else if (isDeferredMsg(message)) {
|
||||
logMsg.message = message();
|
||||
logMsg.stacktrace = stacktrace == null ? '' : stacktrace;
|
||||
} else {
|
||||
// string | object
|
||||
logMsg.message = message;
|
||||
@ -83,7 +119,7 @@ export class Logger {
|
||||
}
|
||||
|
||||
protected sendToAppenders(logMsg: LogMessage) {
|
||||
this.appenders.forEach(app => {
|
||||
this.appenders.forEach((app) => {
|
||||
app.appendMessage(logMsg);
|
||||
});
|
||||
|
||||
|
Reference in New Issue
Block a user