2021-01-10 21:59:07 -05:00
|
|
|
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
|
2018-12-19 19:16:45 +01:00
|
|
|
import { Logger } from "./logger.ts";
|
2020-08-25 09:43:54 +10:00
|
|
|
import type { GenericFunction } from "./logger.ts";
|
2019-01-06 14:19:15 -05:00
|
|
|
import {
|
|
|
|
BaseHandler,
|
|
|
|
ConsoleHandler,
|
2020-03-29 04:03:49 +11:00
|
|
|
FileHandler,
|
2020-04-09 12:45:24 +01:00
|
|
|
RotatingFileHandler,
|
2020-09-27 06:22:32 -04:00
|
|
|
WriterHandler,
|
2019-01-06 14:19:15 -05:00
|
|
|
} from "./handlers.ts";
|
2020-06-07 14:20:33 +01:00
|
|
|
import { assert } from "../_util/assert.ts";
|
2020-07-08 19:26:39 +10:00
|
|
|
import type { LevelName } from "./levels.ts";
|
2020-04-25 02:13:26 -07:00
|
|
|
|
2020-08-12 11:01:36 +01:00
|
|
|
export { LogLevels } from "./levels.ts";
|
|
|
|
export type { LevelName } from "./levels.ts";
|
2020-07-06 13:57:31 +10:00
|
|
|
export { Logger } from "./logger.ts";
|
2018-12-19 19:16:45 +01:00
|
|
|
|
|
|
|
export class LoggerConfig {
|
2020-04-25 02:13:26 -07:00
|
|
|
level?: LevelName;
|
2018-12-19 19:16:45 +01:00
|
|
|
handlers?: string[];
|
|
|
|
}
|
|
|
|
|
2019-01-02 15:12:48 +01:00
|
|
|
export interface LogConfig {
|
2018-12-19 19:16:45 +01:00
|
|
|
handlers?: {
|
2019-01-02 15:12:48 +01:00
|
|
|
[name: string]: BaseHandler;
|
2018-12-19 19:16:45 +01:00
|
|
|
};
|
|
|
|
loggers?: {
|
|
|
|
[name: string]: LoggerConfig;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const DEFAULT_LEVEL = "INFO";
|
2019-01-02 15:12:48 +01:00
|
|
|
const DEFAULT_CONFIG: LogConfig = {
|
2019-01-27 15:21:00 +00:00
|
|
|
handlers: {
|
2020-03-29 04:03:49 +11:00
|
|
|
default: new ConsoleHandler(DEFAULT_LEVEL),
|
2019-01-27 15:21:00 +00:00
|
|
|
},
|
2018-12-19 19:16:45 +01:00
|
|
|
|
|
|
|
loggers: {
|
2019-01-29 01:17:00 +09:00
|
|
|
default: {
|
2019-01-27 15:21:00 +00:00
|
|
|
level: DEFAULT_LEVEL,
|
2020-03-29 04:03:49 +11:00
|
|
|
handlers: ["default"],
|
|
|
|
},
|
|
|
|
},
|
2018-12-19 19:16:45 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const state = {
|
2019-03-05 11:53:35 +11:00
|
|
|
handlers: new Map<string, BaseHandler>(),
|
|
|
|
loggers: new Map<string, Logger>(),
|
2020-03-29 04:03:49 +11:00
|
|
|
config: DEFAULT_CONFIG,
|
2018-12-19 19:16:45 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
export const handlers = {
|
2019-01-02 15:12:48 +01:00
|
|
|
BaseHandler,
|
|
|
|
ConsoleHandler,
|
|
|
|
WriterHandler,
|
2020-03-29 04:03:49 +11:00
|
|
|
FileHandler,
|
2020-04-09 12:45:24 +01:00
|
|
|
RotatingFileHandler,
|
2018-12-19 19:16:45 +01:00
|
|
|
};
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Get a logger instance. If not specified `name`, get the default logger. */
|
2019-03-05 11:53:35 +11:00
|
|
|
export function getLogger(name?: string): Logger {
|
2018-12-19 19:16:45 +01:00
|
|
|
if (!name) {
|
2020-02-07 16:23:38 +09:00
|
|
|
const d = state.loggers.get("default");
|
|
|
|
assert(
|
|
|
|
d != null,
|
2020-07-14 15:24:17 -04:00
|
|
|
`"default" logger must be set for getting logger without name`,
|
2020-02-07 16:23:38 +09:00
|
|
|
);
|
|
|
|
return d;
|
2018-12-19 19:16:45 +01:00
|
|
|
}
|
2020-02-07 16:23:38 +09:00
|
|
|
const result = state.loggers.get(name);
|
|
|
|
if (!result) {
|
2020-06-18 11:50:18 +01:00
|
|
|
const logger = new Logger(name, "NOTSET", { handlers: [] });
|
2019-01-02 15:12:48 +01:00
|
|
|
state.loggers.set(name, logger);
|
|
|
|
return logger;
|
2018-12-19 19:16:45 +01:00
|
|
|
}
|
2020-02-07 16:23:38 +09:00
|
|
|
return result;
|
2018-12-19 19:16:45 +01:00
|
|
|
}
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Log with debug level, using default logger. */
|
2020-06-12 14:27:41 +01:00
|
|
|
export function debug<T>(msg: () => T, ...args: unknown[]): T | undefined;
|
|
|
|
export function debug<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: T extends GenericFunction ? never : T,
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T;
|
|
|
|
export function debug<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: (T extends GenericFunction ? never : T) | (() => T),
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T | undefined {
|
|
|
|
// Assist TS compiler with pass-through generic type
|
|
|
|
if (msg instanceof Function) {
|
|
|
|
return getLogger("default").debug(msg, ...args);
|
|
|
|
}
|
|
|
|
return getLogger("default").debug(msg, ...args);
|
|
|
|
}
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Log with info level, using default logger. */
|
2020-06-12 14:27:41 +01:00
|
|
|
export function info<T>(msg: () => T, ...args: unknown[]): T | undefined;
|
|
|
|
export function info<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: T extends GenericFunction ? never : T,
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T;
|
|
|
|
export function info<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: (T extends GenericFunction ? never : T) | (() => T),
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T | undefined {
|
|
|
|
// Assist TS compiler with pass-through generic type
|
|
|
|
if (msg instanceof Function) {
|
|
|
|
return getLogger("default").info(msg, ...args);
|
|
|
|
}
|
|
|
|
return getLogger("default").info(msg, ...args);
|
|
|
|
}
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Log with warning level, using default logger. */
|
2020-06-12 14:27:41 +01:00
|
|
|
export function warning<T>(msg: () => T, ...args: unknown[]): T | undefined;
|
|
|
|
export function warning<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: T extends GenericFunction ? never : T,
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T;
|
|
|
|
export function warning<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: (T extends GenericFunction ? never : T) | (() => T),
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T | undefined {
|
|
|
|
// Assist TS compiler with pass-through generic type
|
|
|
|
if (msg instanceof Function) {
|
|
|
|
return getLogger("default").warning(msg, ...args);
|
|
|
|
}
|
|
|
|
return getLogger("default").warning(msg, ...args);
|
|
|
|
}
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Log with error level, using default logger. */
|
2020-06-12 14:27:41 +01:00
|
|
|
export function error<T>(msg: () => T, ...args: unknown[]): T | undefined;
|
|
|
|
export function error<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: T extends GenericFunction ? never : T,
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T;
|
|
|
|
export function error<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: (T extends GenericFunction ? never : T) | (() => T),
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T | undefined {
|
|
|
|
// Assist TS compiler with pass-through generic type
|
|
|
|
if (msg instanceof Function) {
|
|
|
|
return getLogger("default").error(msg, ...args);
|
|
|
|
}
|
|
|
|
return getLogger("default").error(msg, ...args);
|
|
|
|
}
|
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Log with critical level, using default logger. */
|
2020-06-12 14:27:41 +01:00
|
|
|
export function critical<T>(msg: () => T, ...args: unknown[]): T | undefined;
|
|
|
|
export function critical<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: T extends GenericFunction ? never : T,
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T;
|
|
|
|
export function critical<T>(
|
2020-08-25 09:43:54 +10:00
|
|
|
msg: (T extends GenericFunction ? never : T) | (() => T),
|
2020-06-12 14:27:41 +01:00
|
|
|
...args: unknown[]
|
|
|
|
): T | undefined {
|
|
|
|
// Assist TS compiler with pass-through generic type
|
|
|
|
if (msg instanceof Function) {
|
|
|
|
return getLogger("default").critical(msg, ...args);
|
|
|
|
}
|
|
|
|
return getLogger("default").critical(msg, ...args);
|
|
|
|
}
|
2019-03-05 11:53:35 +11:00
|
|
|
|
2020-11-06 11:33:59 +09:00
|
|
|
/** Setup logger config. */
|
2019-03-05 11:53:35 +11:00
|
|
|
export async function setup(config: LogConfig): Promise<void> {
|
2019-01-27 15:21:00 +00:00
|
|
|
state.config = {
|
2019-01-29 01:17:00 +09:00
|
|
|
handlers: { ...DEFAULT_CONFIG.handlers, ...config.handlers },
|
2020-03-29 04:03:49 +11:00
|
|
|
loggers: { ...DEFAULT_CONFIG.loggers, ...config.loggers },
|
2019-01-27 15:21:00 +00:00
|
|
|
};
|
2019-01-02 15:12:48 +01:00
|
|
|
|
|
|
|
// tear down existing handlers
|
2019-11-14 05:42:34 +11:00
|
|
|
state.handlers.forEach((handler): void => {
|
|
|
|
handler.destroy();
|
|
|
|
});
|
2019-01-02 15:12:48 +01:00
|
|
|
state.handlers.clear();
|
|
|
|
|
|
|
|
// setup handlers
|
|
|
|
const handlers = state.config.handlers || {};
|
|
|
|
|
|
|
|
for (const handlerName in handlers) {
|
|
|
|
const handler = handlers[handlerName];
|
|
|
|
await handler.setup();
|
|
|
|
state.handlers.set(handlerName, handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
// remove existing loggers
|
|
|
|
state.loggers.clear();
|
|
|
|
|
|
|
|
// setup loggers
|
|
|
|
const loggers = state.config.loggers || {};
|
|
|
|
for (const loggerName in loggers) {
|
|
|
|
const loggerConfig = loggers[loggerName];
|
|
|
|
const handlerNames = loggerConfig.handlers || [];
|
2019-03-05 11:53:35 +11:00
|
|
|
const handlers: BaseHandler[] = [];
|
2019-01-02 15:12:48 +01:00
|
|
|
|
2019-11-14 05:42:34 +11:00
|
|
|
handlerNames.forEach((handlerName): void => {
|
2020-02-07 16:23:38 +09:00
|
|
|
const handler = state.handlers.get(handlerName);
|
|
|
|
if (handler) {
|
|
|
|
handlers.push(handler);
|
2019-01-02 15:12:48 +01:00
|
|
|
}
|
2019-11-14 05:42:34 +11:00
|
|
|
});
|
2019-01-02 15:12:48 +01:00
|
|
|
|
|
|
|
const levelName = loggerConfig.level || DEFAULT_LEVEL;
|
2020-06-18 11:50:18 +01:00
|
|
|
const logger = new Logger(loggerName, levelName, { handlers: handlers });
|
2019-01-02 15:12:48 +01:00
|
|
|
state.loggers.set(loggerName, logger);
|
|
|
|
}
|
2018-12-19 19:16:45 +01:00
|
|
|
}
|
2019-01-02 15:12:48 +01:00
|
|
|
|
2020-05-20 16:27:01 +02:00
|
|
|
await setup(DEFAULT_CONFIG);
|