wallet-core/packages/taler-util/src/logging.ts

223 lines
5.4 KiB
TypeScript
Raw Normal View History

2021-06-08 20:56:27 +02:00
/*
This file is part of TALER
(C) 2019 GNUnet e.V.
TALER is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
TALER is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
TALER; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
/**
* Check if we are running under nodejs.
*/
const isNode =
2021-06-09 15:26:18 +02:00
typeof process !== "undefined" &&
typeof process.release !== "undefined" &&
process.release.name === "node";
2021-06-08 20:56:27 +02:00
export enum LogLevel {
Trace = "trace",
Message = "message",
Info = "info",
Warn = "warn",
Error = "error",
None = "none",
}
export let globalLogLevel = LogLevel.Info;
export function setGlobalLogLevelFromString(logLevelStr: string): void {
globalLogLevel = getLevelForString(logLevelStr);
}
export const byTagLogLevel: Record<string, LogLevel> = {};
export function setLogLevelFromString(tag: string, logLevelStr: string): void {
byTagLogLevel[tag] = getLevelForString(logLevelStr);
}
function getLevelForString(logLevelStr: string): LogLevel {
switch (logLevelStr.toLowerCase()) {
case "trace":
return LogLevel.Trace;
case "info":
return LogLevel.Info;
case "warn":
case "warning":
return LogLevel.Warn;
case "error":
return LogLevel.Error;
case "none":
return LogLevel.None;
default:
if (isNode) {
2021-11-27 20:56:58 +01:00
process.stderr.write(`Invalid log level, defaulting to WARNING\n`);
} else {
console.warn(`Invalid log level, defaulting to WARNING`);
}
return LogLevel.Warn;
}
}
2021-06-08 20:56:27 +02:00
function writeNodeLog(
message: any,
tag: string,
level: string,
args: any[],
): void {
try {
2021-12-01 12:22:58 +01:00
let msg = `${new Date().toISOString()} ${tag} ${level} ${message}`;
2021-06-08 20:56:27 +02:00
if (args.length != 0) {
2021-12-01 12:22:58 +01:00
msg += ` ${JSON.stringify(args, undefined, 2)}\n`;
} else {
msg += `\n`;
2021-06-08 20:56:27 +02:00
}
2021-12-01 12:22:58 +01:00
process.stderr.write(msg);
2021-06-08 20:56:27 +02:00
} catch (e) {
// This can happen when we're trying to log something that doesn't want to be
// converted to a string.
2021-12-01 12:22:58 +01:00
let msg = `${new Date().toISOString()} (logger) FATAL `;
2021-06-08 20:56:27 +02:00
if (e instanceof Error) {
2021-12-01 12:22:58 +01:00
msg += `failed to write log: ${e.message}\n`;
} else {
msg += "failed to write log\n";
2021-06-08 20:56:27 +02:00
}
2021-12-01 12:22:58 +01:00
process.stderr.write(msg);
2021-06-08 20:56:27 +02:00
}
}
/**
* Logger that writes to stderr when running under node,
* and uses the corresponding console.* method to log in the browser.
*/
export class Logger {
2023-01-31 15:42:12 +01:00
constructor(private tag: string) {}
2021-06-08 20:56:27 +02:00
shouldLogTrace(): boolean {
const level = byTagLogLevel[this.tag] ?? globalLogLevel;
switch (level) {
case LogLevel.Trace:
return true;
case LogLevel.Message:
case LogLevel.Info:
case LogLevel.Warn:
case LogLevel.Error:
case LogLevel.None:
return false;
}
2021-08-19 16:06:09 +02:00
}
shouldLogInfo(): boolean {
const level = byTagLogLevel[this.tag] ?? globalLogLevel;
switch (level) {
case LogLevel.Trace:
case LogLevel.Message:
case LogLevel.Info:
return true;
case LogLevel.Warn:
case LogLevel.Error:
case LogLevel.None:
return false;
}
2021-08-19 16:06:09 +02:00
}
shouldLogWarn(): boolean {
const level = byTagLogLevel[this.tag] ?? globalLogLevel;
switch (level) {
case LogLevel.Trace:
case LogLevel.Message:
case LogLevel.Info:
case LogLevel.Warn:
return true;
case LogLevel.Error:
case LogLevel.None:
return false;
}
}
shouldLogError(): boolean {
const level = byTagLogLevel[this.tag] ?? globalLogLevel;
switch (level) {
case LogLevel.Trace:
case LogLevel.Message:
case LogLevel.Info:
case LogLevel.Warn:
case LogLevel.Error:
2021-11-27 20:56:58 +01:00
return true;
case LogLevel.None:
return false;
}
2021-08-19 16:06:09 +02:00
}
2021-06-08 20:56:27 +02:00
info(message: string, ...args: any[]): void {
if (!this.shouldLogInfo()) {
return;
}
2021-06-08 20:56:27 +02:00
if (isNode) {
writeNodeLog(message, this.tag, "INFO", args);
} else {
console.info(
`${new Date().toISOString()} ${this.tag} INFO ` + message,
...args,
);
}
}
warn(message: string, ...args: any[]): void {
if (!this.shouldLogWarn()) {
return;
}
2021-06-08 20:56:27 +02:00
if (isNode) {
writeNodeLog(message, this.tag, "WARN", args);
} else {
console.warn(
`${new Date().toISOString()} ${this.tag} INFO ` + message,
...args,
);
}
}
error(message: string, ...args: any[]): void {
if (!this.shouldLogError()) {
return;
}
2021-06-08 20:56:27 +02:00
if (isNode) {
writeNodeLog(message, this.tag, "ERROR", args);
} else {
console.info(
`${new Date().toISOString()} ${this.tag} ERROR ` + message,
...args,
);
}
}
trace(message: string, ...args: any[]): void {
if (!this.shouldLogTrace()) {
return;
}
2021-06-08 20:56:27 +02:00
if (isNode) {
writeNodeLog(message, this.tag, "TRACE", args);
} else {
console.info(
`${new Date().toISOString()} ${this.tag} TRACE ` + message,
...args,
);
}
}
reportBreak(): void {
if (!this.shouldLogError()) {
return;
}
const location = new Error("programming error");
this.error(`assertion failed: ${location.stack}`);
}
2021-06-08 20:56:27 +02:00
}