Compare commits
15 Commits
b92caf6468
...
4.0.2
Author | SHA1 | Date | |
---|---|---|---|
3526766e68 | |||
3de5f368ef | |||
e0b51625d8 | |||
153aca0ccb | |||
7ca0c4fd72 | |||
8e183ac1a5 | |||
feed4626e6 | |||
9d4e521619 | |||
7d75f65dd3 | |||
eeed068ddd | |||
6daf815ea8 | |||
357b98c69a | |||
96d7808f35 | |||
176d37249d | |||
bcff79fc90 |
2
.vscode/launch.json
vendored
2
.vscode/launch.json
vendored
@ -9,7 +9,7 @@
|
||||
"request": "launch",
|
||||
"name": "Launch Program",
|
||||
"skipFiles": ["<node_internals>/**"],
|
||||
"program": "${workspaceFolder}\\out\\test.js",
|
||||
"program": "${workspaceFolder}/out/test.js",
|
||||
"preLaunchTask": "tsc: build - tsconfig.json",
|
||||
"outFiles": ["${workspaceFolder}/out/**/*.js"]
|
||||
}
|
||||
|
@ -1,29 +1,29 @@
|
||||
# Logging
|
||||
|
||||
Simple logging module, that supports terminal coloring and different plugins
|
||||
|
||||
# Getting Started
|
||||
## Getting Started
|
||||
|
||||
``` javascript
|
||||
|
||||
const Logging = require("@hibas123/logging").Logging;
|
||||
```javascript
|
||||
const Logging = require("@hibas123/logging").default;
|
||||
|
||||
Logging.log("Hello there");
|
||||
|
||||
```
|
||||
|
||||
There are different Logging levels, that also apply terminal coloring:
|
||||
|
||||
``` javascript
|
||||
Logging.debug("Debug message")
|
||||
Logging.log("Log message")
|
||||
Logging.warning("Warning")
|
||||
Logging.error(new Error("To less creativity"))
|
||||
Logging.error("Just an simple message as error")
|
||||
Logging.errorMessage("Nearly the same as error")
|
||||
```javascript
|
||||
Logging.debug("Debug message");
|
||||
Logging.log("Log message");
|
||||
Logging.warning("Warning");
|
||||
Logging.error(new Error("To less creativity"));
|
||||
Logging.error("Just an simple message as error");
|
||||
Logging.errorMessage("Nearly the same as error");
|
||||
```
|
||||
|
||||
All Logging types except the simple error take as many arguments as you want. These will be joined with spaces and serialized with the node util.inspect function.
|
||||
|
||||
# Setup
|
||||
## Setup
|
||||
|
||||
This logging module doesn't require any setup per default, but sometimes it makes sense to configure a few things.
|
||||
|
||||
@ -31,28 +31,32 @@ For example can you disable the console output. This may be helpful, if you inse
|
||||
|
||||
Also you can set a name. All messages that are send with this instance are prefixed by this name.
|
||||
|
||||
``` javascript
|
||||
const CustomLogging = new LoggingBase(name | {
|
||||
```javascript
|
||||
const CustomLogging = new LoggingBase(
|
||||
name |
|
||||
{
|
||||
name: "custom", // default undefined
|
||||
console: false // default true
|
||||
});
|
||||
console: false, // default true
|
||||
}
|
||||
);
|
||||
```
|
||||
|
||||
|
||||
# Plugins
|
||||
## Plugins
|
||||
|
||||
There is a Plugin API available, that makes is possible to add custom Logging Adapter.
|
||||
|
||||
``` javascript
|
||||
```javascript
|
||||
const Demo = new LoggingBase("Demo");
|
||||
Demo.addAdapter(new DemoAdapter({ color: "rainbow" }));
|
||||
```
|
||||
|
||||
The adapters need to provide a very simple Interface:
|
||||
|
||||
``` typescript
|
||||
```typescript
|
||||
interface Adapter {
|
||||
init(observable: ObservableInterface<Message>, name?: string): void | Promise<void>;
|
||||
init(): void | Promise<void>;
|
||||
|
||||
onMessage(message: Message): void;
|
||||
|
||||
flush(sync: true): void;
|
||||
flush(sync: false): void | Promise<void>;
|
||||
@ -60,28 +64,33 @@ interface Adapter {
|
||||
|
||||
interface Message {
|
||||
type: LoggingTypes;
|
||||
name?:string;
|
||||
text: {
|
||||
raw: string[],
|
||||
formatted: string[]
|
||||
};
|
||||
names?: string[];
|
||||
text: IFormatted;
|
||||
date: Date;
|
||||
file: string;
|
||||
}
|
||||
|
||||
export interface IFormatted {
|
||||
color?: IColors;
|
||||
bgcolor?: IColors;
|
||||
bold?: boolean;
|
||||
italic?: boolean;
|
||||
blink?: boolean;
|
||||
underscore?: boolean;
|
||||
|
||||
content: string;
|
||||
}
|
||||
|
||||
enum LoggingTypes {
|
||||
Log,
|
||||
Warning,
|
||||
Error,
|
||||
Debug
|
||||
Debug,
|
||||
}
|
||||
```
|
||||
|
||||
The `ObservableInterface` comes from `@hibas123/utils`. It provides a very simple api for subscribing and unsubscribing from the message events.
|
||||
## License
|
||||
|
||||
More Details on Observable [git](https://git.stamm.me/OpenServer/Utils) or [npm](https://www.npmjs.com/package/@hibas123/utils)
|
||||
|
||||
# License
|
||||
MIT
|
||||
|
||||
Copyright (c) 2018 Fabian Stamm
|
54
benchmark.js
Normal file
54
benchmark.js
Normal file
@ -0,0 +1,54 @@
|
||||
const { LoggingBase } = require("./out/index.js");
|
||||
let results = {};
|
||||
|
||||
function benchmark(name, count, runner) {
|
||||
console.profile(name);
|
||||
const start = process.hrtime.bigint()
|
||||
|
||||
runner(count);
|
||||
|
||||
const diffNS = process.hrtime.bigint() - start;
|
||||
const diffMS = Number(diffNS / 1000n / 1000n);
|
||||
console.profileEnd(name)
|
||||
|
||||
results[name]= {
|
||||
count,
|
||||
time: diffMS,
|
||||
timePerI: (diffMS / count).toFixed(4)
|
||||
}
|
||||
}
|
||||
|
||||
benchmark("simple", 10000000, (cnt) => {
|
||||
const l = new LoggingBase({
|
||||
console: false
|
||||
});
|
||||
for (let i = 0; i < cnt; i++) {
|
||||
l.log("simple log")
|
||||
}
|
||||
})
|
||||
|
||||
benchmark("complex", 1000000, (cnt) => {
|
||||
const l = new LoggingBase({
|
||||
console: false
|
||||
});
|
||||
for (let i = 0; i < cnt; i++) {
|
||||
l.log("complex log", {
|
||||
a: 1,
|
||||
b: {
|
||||
c:"test"
|
||||
}
|
||||
})
|
||||
}
|
||||
})
|
||||
|
||||
benchmark("very long", 10000000, (cnt) => {
|
||||
const l = new LoggingBase({
|
||||
console: false
|
||||
});
|
||||
const longText = "complex log".repeat(100);
|
||||
for (let i = 0; i < cnt; i++) {
|
||||
l.log(longText)
|
||||
}
|
||||
})
|
||||
|
||||
console.table(results)
|
8
deno_build.js
Normal file
8
deno_build.js
Normal file
@ -0,0 +1,8 @@
|
||||
const pkg = JSON.parse(await Deno.readTextFile("package.json"));
|
||||
const meta = JSON.parse(await Deno.readTextFile("meta.json"));
|
||||
|
||||
meta.version = pkg.version;
|
||||
|
||||
await Deno.copyFile("README.md", "esm/README.md");
|
||||
|
||||
await Deno.writeTextFile("meta.json", JSON.stringify(meta, undefined, 3));
|
16
meta.json
Normal file
16
meta.json
Normal file
@ -0,0 +1,16 @@
|
||||
{
|
||||
"name": "logging",
|
||||
"version": "3.1.2",
|
||||
"description": "",
|
||||
"author": "Fabian Stamm <dev@fabianstamm.de>",
|
||||
"contributors": [],
|
||||
"hooks": {
|
||||
"prepublish": "deno_build.js"
|
||||
},
|
||||
"root": "esm",
|
||||
"files": [
|
||||
"esm/**/*.ts",
|
||||
"esm/**/*.js",
|
||||
"esm/README.md"
|
||||
]
|
||||
}
|
5464
package-lock.json
generated
5464
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
28
package.json
28
package.json
@ -1,14 +1,17 @@
|
||||
{
|
||||
"name": "@hibas123/logging",
|
||||
"version": "2.5.4",
|
||||
"version": "4.0.2",
|
||||
"description": "",
|
||||
"main": "out/index.js",
|
||||
"types": "out/index.d.ts",
|
||||
"type": "module",
|
||||
"main": "esm/index.js",
|
||||
"types": "esm/index.d.ts",
|
||||
"module": "esm/index.js",
|
||||
"scripts": {
|
||||
"prepublish": "npm run build",
|
||||
"build": "tsc && tsc -p tsconfig.esm.json",
|
||||
"dev": "nodemon -e ts --exec ts-node src/test.ts"
|
||||
"prepublishOnly": "npm run build",
|
||||
"build": "tsc",
|
||||
"test": "tsc && node esm/test.js",
|
||||
"bench": "tsc && node benchmark.js",
|
||||
"prof": "tsc && ndb --prof benchmark.js"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
@ -18,18 +21,15 @@
|
||||
"license": "MIT",
|
||||
"files": [
|
||||
"src/",
|
||||
"out/",
|
||||
"esm/",
|
||||
"tsconfig.json",
|
||||
"readme.md"
|
||||
],
|
||||
"devDependencies": {
|
||||
"concurrently": "^5.3.0",
|
||||
"nodemon": "^2.0.4",
|
||||
"ts-node": "^9.0.0",
|
||||
"typescript": "^4.0.2"
|
||||
},
|
||||
"dependencies": {
|
||||
"@hibas123/utils": "^2.2.10"
|
||||
"concurrently": "^9.2.1",
|
||||
"ndb": "^1.1.5",
|
||||
"nodemon": "^3.1.10",
|
||||
"ts-node": "^10.9.2",
|
||||
"typescript": "^5.9.3"
|
||||
}
|
||||
}
|
668
src/base.ts
668
src/base.ts
@ -1,36 +1,19 @@
|
||||
import { Observable, ObservableInterface } from "@hibas123/utils";
|
||||
import { ConsoleAdapter } from "./consolewriter.js";
|
||||
import inspect from "./inspect.js";
|
||||
import {
|
||||
Adapter,
|
||||
LoggingTypes,
|
||||
Message,
|
||||
FormatConfig,
|
||||
DefaultFormatConfig,
|
||||
Format,
|
||||
FormatTypes,
|
||||
Colors,
|
||||
FormattedText,
|
||||
FormattedLine,
|
||||
FormatConfig,
|
||||
Formatted,
|
||||
ILoggingInterface,
|
||||
LoggingTypes,
|
||||
Message,
|
||||
} from "./types.js";
|
||||
|
||||
const browser = typeof window !== "undefined";
|
||||
|
||||
export function removeColors(text: string) {
|
||||
text = text.replace(
|
||||
/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g,
|
||||
""
|
||||
);
|
||||
|
||||
// let index = text.indexOf("\x1b");
|
||||
// while (index >= 0) {
|
||||
// text = text.substring(0, index) + text.substring(index + 5, text.length);
|
||||
// index = text.indexOf("\x1b");
|
||||
// }
|
||||
return text;
|
||||
}
|
||||
|
||||
export interface LoggingBaseOptions {
|
||||
export interface ILoggingOptions {
|
||||
/**
|
||||
* Name will be prefixed on Console output and added to logfiles, if not specified here
|
||||
*/
|
||||
@ -39,290 +22,263 @@ export interface LoggingBaseOptions {
|
||||
* Prints output to console
|
||||
*/
|
||||
console: boolean;
|
||||
}
|
||||
|
||||
const adapterCache = new WeakMap<Adapter, number>();
|
||||
|
||||
class AdapterSet {
|
||||
change = new Observable<{ type: "add" | "remove"; adapter: Adapter }>();
|
||||
adapters: Set<Adapter> = new Set();
|
||||
|
||||
addAdapter(adapter: Adapter) {
|
||||
if (!this.adapters.has(adapter)) {
|
||||
this.adapters.add(adapter);
|
||||
this.change.send({
|
||||
type: "add",
|
||||
adapter: adapter,
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const consoleAdapter = new ConsoleAdapter();
|
||||
|
||||
declare var process: { cwd: () => string };
|
||||
|
||||
const PROJECT_ROOT = typeof process !== "undefined" ? process.cwd() : undefined;
|
||||
|
||||
export class LoggingBase {
|
||||
private _formatMap: FormatConfig = new DefaultFormatConfig();
|
||||
|
||||
public set formatMap(value: FormatConfig) {
|
||||
this._formatMap = value;
|
||||
}
|
||||
|
||||
private adapterSet: AdapterSet;
|
||||
private adapter_init: Promise<void>[] = [];
|
||||
|
||||
private timerMap = new Map<string, { name: string; start: any }>();
|
||||
|
||||
private messageObservable = new Observable<Message>();
|
||||
protected _name: string;
|
||||
|
||||
private _logLevel = LoggingTypes.Debug;
|
||||
|
||||
get logLevel() {
|
||||
return this._logLevel;
|
||||
}
|
||||
|
||||
set logLevel(value: LoggingTypes) {
|
||||
this._logLevel = value;
|
||||
}
|
||||
|
||||
get name() {
|
||||
return this._name;
|
||||
}
|
||||
|
||||
constructor(
|
||||
options?: Partial<LoggingBaseOptions> | string,
|
||||
adapterSet?: AdapterSet
|
||||
) {
|
||||
let opt: Partial<LoggingBaseOptions>;
|
||||
if (!options) opt = {};
|
||||
else if (typeof options === "string") {
|
||||
opt = { name: options };
|
||||
} else {
|
||||
opt = options;
|
||||
}
|
||||
|
||||
let config: LoggingBaseOptions = {
|
||||
name: undefined,
|
||||
console: true,
|
||||
...opt,
|
||||
};
|
||||
|
||||
if (config.name) this._name = config.name;
|
||||
|
||||
for (let key in this) {
|
||||
if (typeof this[key] === "function")
|
||||
this[key] = (<any>this[key]).bind(this);
|
||||
}
|
||||
|
||||
if (adapterSet) {
|
||||
this.adapterSet = adapterSet;
|
||||
this.adapterSet.adapters.forEach((a) => this.initAdapter(a));
|
||||
} else {
|
||||
this.adapterSet = new AdapterSet();
|
||||
}
|
||||
|
||||
this.adapterSet.change.subscribe((change) => {
|
||||
if (change.type === "add") {
|
||||
this.initAdapter(change.adapter);
|
||||
}
|
||||
});
|
||||
|
||||
if (config.console) {
|
||||
this.addAdapter(consoleAdapter);
|
||||
}
|
||||
|
||||
//Binding function to this
|
||||
this.debug = this.debug.bind(this);
|
||||
this.log = this.log.bind(this);
|
||||
this.warn = this.warn.bind(this);
|
||||
this.warning = this.warning.bind(this);
|
||||
this.error = this.error.bind(this);
|
||||
this.errorMessage = this.errorMessage.bind(this);
|
||||
this.flush = this.flush.bind(this);
|
||||
}
|
||||
|
||||
/**
|
||||
* Can be used to override function from super class
|
||||
* @param child New child logging instance
|
||||
* Enables printing of calling file
|
||||
*/
|
||||
protected postGetChild(child: LoggingBase) {}
|
||||
resolve_filename: boolean;
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a new logging instance, with the adapters liked together.
|
||||
* @param name Name/Prefix of the new child. The actual name will resolve as "<parent-name>/<name>"
|
||||
*/
|
||||
getChild(name: string) {
|
||||
let lg = new LoggingBase(
|
||||
{
|
||||
console: false,
|
||||
name: this.name ? this.name + "/" + name : name,
|
||||
},
|
||||
this.adapterSet
|
||||
);
|
||||
return lg;
|
||||
}
|
||||
export interface INativeFunctions {
|
||||
startTimer(): any;
|
||||
diffTime(start: any, end: any): number;
|
||||
endTimer(start: any): number;
|
||||
}
|
||||
|
||||
private initAdapter(adapter: Adapter) {
|
||||
let cached = adapterCache.get(adapter) || 0;
|
||||
adapterCache.set(adapter, cached + 1);
|
||||
|
||||
let prms = Promise.resolve(
|
||||
adapter.init(this.messageObservable.getPublicApi())
|
||||
);
|
||||
this.adapter_init.push(prms);
|
||||
}
|
||||
|
||||
addAdapter(adapter: Adapter) {
|
||||
this.adapterSet.addAdapter(adapter);
|
||||
}
|
||||
|
||||
flush(sync: true): void;
|
||||
flush(sync: false): Promise<void>;
|
||||
flush(sync: boolean): void | Promise<void> {
|
||||
if (sync) {
|
||||
this.adapterSet.adapters.forEach((elm) => elm.flush(true));
|
||||
} else {
|
||||
let adapters: (void | Promise<void>)[] = [];
|
||||
this.adapterSet.adapters.forEach((elm) =>
|
||||
adapters.push(elm.flush(false))
|
||||
);
|
||||
return Promise.all(adapters).then(() => {});
|
||||
}
|
||||
}
|
||||
|
||||
#closed = false;
|
||||
|
||||
public close() {
|
||||
if (this.#closed) return;
|
||||
this.#closed = true;
|
||||
|
||||
this.adapterSet.adapters.forEach((adapter) => {
|
||||
let cached = adapterCache.get(adapter);
|
||||
|
||||
if (cached) {
|
||||
cached--;
|
||||
if (cached <= 0) {
|
||||
adapterCache.delete(adapter);
|
||||
adapter.close();
|
||||
} else adapterCache.set(adapter, cached);
|
||||
}
|
||||
|
||||
adapter.close ? adapter.close() : undefined;
|
||||
});
|
||||
|
||||
this.adapterSet = undefined;
|
||||
this.messageObservable.close();
|
||||
}
|
||||
|
||||
public waitForSetup() {
|
||||
return Promise.all(this.adapter_init);
|
||||
}
|
||||
|
||||
debug(...message: any[]) {
|
||||
if (this._logLevel <= LoggingTypes.Debug)
|
||||
this.message(LoggingTypes.Debug, message);
|
||||
}
|
||||
|
||||
log(...message: any[]) {
|
||||
if (this._logLevel <= LoggingTypes.Log)
|
||||
this.message(LoggingTypes.Log, message);
|
||||
}
|
||||
|
||||
warning(...message: any[]) {
|
||||
if (this._logLevel <= LoggingTypes.Warning)
|
||||
this.message(LoggingTypes.Warning, message);
|
||||
}
|
||||
|
||||
warn(...message: any[]) {
|
||||
if (this._logLevel <= LoggingTypes.Warning)
|
||||
this.message(LoggingTypes.Warning, message);
|
||||
}
|
||||
|
||||
error(error: Error | string) {
|
||||
if (this._logLevel > LoggingTypes.Error) return;
|
||||
if (!error)
|
||||
error = "Empty ERROR was passed, so no informations available";
|
||||
if (typeof error === "string") {
|
||||
let e = new Error("This is a fake error, to get a stack trace");
|
||||
this.message(LoggingTypes.Error, [error, "\n", e.stack]);
|
||||
} else {
|
||||
this.message(
|
||||
LoggingTypes.Error,
|
||||
[error.message, "\n", error.stack],
|
||||
getCallerFromExisting(error)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
errorMessage(...message: any[]) {
|
||||
if (this._logLevel <= LoggingTypes.Error)
|
||||
this.message(LoggingTypes.Error, message);
|
||||
}
|
||||
|
||||
protected getCurrentTime(): any {
|
||||
export const DefaultNativeFunctions = {
|
||||
startTimer: () => {
|
||||
if (browser && window.performance && window.performance.now) {
|
||||
return window.performance.now();
|
||||
} else {
|
||||
return Date.now();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* The time difference in milliseconds (fractions allowed!)
|
||||
* @param start Start time from getCurrentTime
|
||||
*/
|
||||
protected getTimeDiff(start: any) {
|
||||
},
|
||||
diffTime: (start: any, end: any) => {
|
||||
return end - start;
|
||||
},
|
||||
endTimer: (start: any) => {
|
||||
if (browser && window.performance && window.performance.now) {
|
||||
return window.performance.now() - start;
|
||||
} else {
|
||||
return Date.now() - start;
|
||||
}
|
||||
},
|
||||
} as INativeFunctions;
|
||||
|
||||
const consoleAdapter = new ConsoleAdapter();
|
||||
declare var process: { cwd: () => string };
|
||||
const PROJECT_ROOT = typeof process !== "undefined" ? process.cwd() : undefined;
|
||||
|
||||
const InitialisedAdapters = Symbol("@hibas123/logging:initialisedAdapters");
|
||||
|
||||
export abstract class LoggingInterface implements ILoggingInterface {
|
||||
#names: string[];
|
||||
#timerMap = new Map<string, { name: string; start: any, marks: { time: any, name: string }[] }>();
|
||||
|
||||
get names() {
|
||||
return [...this.#names];
|
||||
}
|
||||
|
||||
time(id?: string, name = id) {
|
||||
if (!id) {
|
||||
id = Math.floor(Math.random() * 899999 + 100000).toString();
|
||||
protected abstract message(
|
||||
type: LoggingTypes,
|
||||
names: string[],
|
||||
message: any[],
|
||||
caller?: { file: string; line: number; column?: number }
|
||||
): void;
|
||||
|
||||
constructor(names: string[]) {
|
||||
this.#names = names;
|
||||
|
||||
for (const key in this) {
|
||||
if (typeof this[key] === "function") {
|
||||
this[key] = (this[key] as never as Function).bind(this);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
this.timerMap.set(id, {
|
||||
name,
|
||||
start: this.getCurrentTime(),
|
||||
});
|
||||
debug(...message: any[]) {
|
||||
this.message(LoggingTypes.Debug, this.#names, message);
|
||||
}
|
||||
|
||||
log(...message: any[]) {
|
||||
this.message(LoggingTypes.Log, this.#names, message);
|
||||
}
|
||||
|
||||
warning(...message: any[]) {
|
||||
this.message(LoggingTypes.Warn, this.#names, message);
|
||||
}
|
||||
|
||||
warn(...message: any[]) {
|
||||
this.message(LoggingTypes.Warn, this.#names, message);
|
||||
}
|
||||
|
||||
error(error: Error | string, ...message: any[]) {
|
||||
if (!error)
|
||||
error = "Empty ERROR was passed, so no informations available";
|
||||
if (typeof error === "string") {
|
||||
let e = new Error("This is a fake error, to get a stack trace");
|
||||
this.message(LoggingTypes.Error, this.#names, [
|
||||
error,
|
||||
...message,
|
||||
"\n",
|
||||
e.stack,
|
||||
"\n",
|
||||
]);
|
||||
} else {
|
||||
this.message(
|
||||
LoggingTypes.Error,
|
||||
this.#names,
|
||||
[error.message, "\n", error.stack, "\n", ...message],
|
||||
getCallerFromExisting(error)
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
time(id?: string) {
|
||||
if (!id) id = Math.floor(Math.random() * 899999 + 100000).toString();
|
||||
|
||||
let timer = {
|
||||
name: id,
|
||||
start: LoggingBase.nativeFunctions.startTimer(),
|
||||
marks: [],
|
||||
};
|
||||
this.#timerMap.set(id, timer);
|
||||
|
||||
return {
|
||||
id,
|
||||
mark: (label: string) => {
|
||||
timer.marks.push({ time: LoggingBase.nativeFunctions.startTimer(), name: label });
|
||||
},
|
||||
end: () => this.timeEnd(id),
|
||||
};
|
||||
}
|
||||
|
||||
timeEnd(id: string) {
|
||||
let timer = this.timerMap.get(id);
|
||||
let timer = this.#timerMap.get(id);
|
||||
if (timer) {
|
||||
let diff = this.getTimeDiff(timer.start);
|
||||
this.message(LoggingTypes.Debug, [
|
||||
withColor(Colors.GREEN, `[${timer.name}]`),
|
||||
let end_time = LoggingBase.nativeFunctions.startTimer();
|
||||
let diff = LoggingBase.nativeFunctions.diffTime(timer.start, end_time);
|
||||
|
||||
this.message(LoggingTypes.Debug, this.#names, [
|
||||
Format.green(`[${timer.name}]`),
|
||||
`->`,
|
||||
withColor(Colors.BLUE, diff.toFixed(4)),
|
||||
Format.blue(diff.toFixed(4)),
|
||||
"ms",
|
||||
]);
|
||||
|
||||
if (timer.marks.length > 0) {
|
||||
let last_mark = timer.start;
|
||||
for (let mark of timer.marks) {
|
||||
let diff = LoggingBase.nativeFunctions.diffTime(last_mark, mark.time);
|
||||
this.message(LoggingTypes.Debug, [...this.#names, timer.name], [
|
||||
Format.green(`[${mark.name}]`),
|
||||
`->`,
|
||||
Format.blue(diff.toFixed(4)),
|
||||
"ms",
|
||||
]);
|
||||
last_mark = mark.time;
|
||||
}
|
||||
let diff = LoggingBase.nativeFunctions.diffTime(last_mark, end_time);
|
||||
this.message(LoggingTypes.Debug, [...this.#names, timer.name], [
|
||||
Format.green(`[end]`),
|
||||
`->`,
|
||||
Format.blue(diff.toFixed(4)),
|
||||
"ms",
|
||||
]);
|
||||
}
|
||||
|
||||
return diff;
|
||||
}
|
||||
return -1;
|
||||
}
|
||||
|
||||
private message(
|
||||
abstract getChild(name: string): ILoggingInterface;
|
||||
}
|
||||
|
||||
export class LoggingBase extends LoggingInterface {
|
||||
private static [InitialisedAdapters] = new Map<Adapter, number>();
|
||||
public static nativeFunctions: INativeFunctions = DefaultNativeFunctions;
|
||||
|
||||
static DecoupledLogging = class extends LoggingInterface {
|
||||
#lg: LoggingBase;
|
||||
constructor(names: string[], lg: LoggingBase) {
|
||||
super(names);
|
||||
this.#lg = lg;
|
||||
}
|
||||
|
||||
message(...params: [any, any, any, any]) {
|
||||
this.#lg.message(...params);
|
||||
}
|
||||
|
||||
getChild(name: string) {
|
||||
return new LoggingBase.DecoupledLogging(
|
||||
[...this.names, name],
|
||||
this.#lg
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
#closed = false;
|
||||
#logLevel = LoggingTypes.Debug;
|
||||
#resolve_filename: boolean;
|
||||
#format_map: FormatConfig = new DefaultFormatConfig();
|
||||
#adapters = new Set<Adapter>();
|
||||
|
||||
set logLevel(level: LoggingTypes) {
|
||||
this.#logLevel = level;
|
||||
}
|
||||
|
||||
get logLevel() {
|
||||
return this.#logLevel;
|
||||
}
|
||||
|
||||
constructor(options?: Partial<ILoggingOptions>) {
|
||||
super(options?.name ? [options.name] : []);
|
||||
options = {
|
||||
console: true,
|
||||
resolve_filename: true,
|
||||
...options,
|
||||
};
|
||||
|
||||
if (options.console) {
|
||||
this.addAdapter(consoleAdapter);
|
||||
}
|
||||
}
|
||||
|
||||
async addAdapter(adapter: Adapter) {
|
||||
const init = adapter.init();
|
||||
|
||||
const add = () => {
|
||||
this.#adapters.add(adapter);
|
||||
if (LoggingBase[InitialisedAdapters].has(adapter)) {
|
||||
LoggingBase[InitialisedAdapters].set(
|
||||
adapter,
|
||||
LoggingBase[InitialisedAdapters].get(adapter) + 1
|
||||
);
|
||||
} else {
|
||||
LoggingBase[InitialisedAdapters].set(adapter, 1);
|
||||
}
|
||||
};
|
||||
|
||||
if (!init) {
|
||||
add();
|
||||
} else {
|
||||
await Promise.resolve(init).then(add);
|
||||
}
|
||||
}
|
||||
|
||||
getChild(name: string): ILoggingInterface {
|
||||
return new LoggingBase.DecoupledLogging([...this.names, name], this);
|
||||
}
|
||||
|
||||
protected message(
|
||||
type: LoggingTypes,
|
||||
names: string[],
|
||||
message: any[],
|
||||
caller?: { file: string; line: number; column?: number }
|
||||
) {
|
||||
if (this.#closed) return;
|
||||
if (this.#logLevel > type) return;
|
||||
if (this.#closed) {
|
||||
//TODO: Maybe error?
|
||||
return;
|
||||
}
|
||||
|
||||
let date = new Date().toISOString().replace(/T/, " ").replace(/\..+/, "");
|
||||
const date = new Date();
|
||||
const isoStr = date.toISOString();
|
||||
const date_str = isoStr.substring(0, 10) + " " + isoStr.substring(11, 19);
|
||||
|
||||
let file: string | undefined = undefined;
|
||||
if (this.#resolve_filename) {
|
||||
let file_raw = caller;
|
||||
if (!file_raw) {
|
||||
try {
|
||||
@ -348,121 +304,117 @@ export class LoggingBase {
|
||||
|
||||
file_raw.file = newF;
|
||||
}
|
||||
let file = `${file_raw.file || "<unknown>"}:${file_raw.line}:${
|
||||
file_raw.column || 0
|
||||
file = `${file_raw.file || "<unknown>"}:${file_raw.line}:${file_raw.column || 0
|
||||
}`;
|
||||
}
|
||||
|
||||
let type_str = LoggingTypes[type].toUpperCase().padEnd(5, " ");
|
||||
let type_format: Format[] = [];
|
||||
let type_format: Formatted;
|
||||
switch (type) {
|
||||
case LoggingTypes.Log:
|
||||
type_format = this._formatMap.log;
|
||||
type_format = this.#format_map.log;
|
||||
break;
|
||||
case LoggingTypes.Error:
|
||||
type_format = this._formatMap.error;
|
||||
type_format = this.#format_map.error;
|
||||
break;
|
||||
case LoggingTypes.Debug:
|
||||
type_format = this._formatMap.debug;
|
||||
type_format = this.#format_map.debug;
|
||||
break;
|
||||
case LoggingTypes.Warning:
|
||||
type_format = this._formatMap.warning;
|
||||
case LoggingTypes.Warn:
|
||||
type_format = this.#format_map.warning;
|
||||
break;
|
||||
}
|
||||
|
||||
const prefix: FormattedText[] = [];
|
||||
const a = (text: string, formats: Format[] = []) => {
|
||||
prefix.push({
|
||||
text,
|
||||
formats,
|
||||
});
|
||||
};
|
||||
|
||||
a("[");
|
||||
a(date, this._formatMap.date);
|
||||
a("][");
|
||||
a(type_str, type_format);
|
||||
if (file_raw.file) {
|
||||
a("][");
|
||||
a(file, this._formatMap.file);
|
||||
}
|
||||
a("]: ");
|
||||
|
||||
let raw: string[] = [];
|
||||
|
||||
const formatted: FormattedLine[] = [];
|
||||
let line: FormattedLine;
|
||||
const newLine = () => {
|
||||
if (line && line.length > 0) {
|
||||
formatted.push(line);
|
||||
raw.push(line.map((e) => e.text).join(""));
|
||||
}
|
||||
line = [...prefix];
|
||||
};
|
||||
newLine();
|
||||
|
||||
message.forEach((e, i) => {
|
||||
let formats: Format[] = [];
|
||||
if (typeof e !== "string") {
|
||||
if (e && typeof e === "object") {
|
||||
if (e[colorSymbol]) {
|
||||
formats.push({
|
||||
type: FormatTypes.COLOR,
|
||||
color: e[colorSymbol],
|
||||
});
|
||||
e = e.value;
|
||||
const nameFormatted: Formatted<string>[] = [];
|
||||
if (names.length > 0) {
|
||||
nameFormatted.push(new Formatted("["));
|
||||
for (let i = 0; i < names.length; i++) {
|
||||
nameFormatted.push(new Formatted(names[i], this.#format_map.names));
|
||||
if (i < names.length - 1) {
|
||||
nameFormatted.push(this.#format_map.names_delimiter);
|
||||
}
|
||||
}
|
||||
if (typeof e !== "string")
|
||||
e = inspect(e, {
|
||||
colors: true,
|
||||
nameFormatted.push(new Formatted("]"));
|
||||
}
|
||||
|
||||
let linePrefix = [
|
||||
new Formatted("["),
|
||||
new Formatted(date_str, this.#format_map.date),
|
||||
new Formatted("]["),
|
||||
new Formatted(type_str, type_format),
|
||||
...(file
|
||||
? [new Formatted("]["), new Formatted(file, this.#format_map.file)]
|
||||
: []),
|
||||
new Formatted("]"),
|
||||
...nameFormatted,
|
||||
new Formatted(": "),
|
||||
];
|
||||
|
||||
let formattedMessage: Formatted<string>[] = [...linePrefix];
|
||||
message.forEach((msg, idx) => {
|
||||
let format: Formatted;
|
||||
if (msg instanceof Formatted) {
|
||||
format = msg;
|
||||
msg = msg.content;
|
||||
} else {
|
||||
format = new Formatted();
|
||||
}
|
||||
|
||||
if (typeof msg !== "string") {
|
||||
msg = inspect(msg, {
|
||||
colors: false, //TODO: Maybe change when changing the removeColors to return formatted text?
|
||||
showHidden: true,
|
||||
depth: 3,
|
||||
}) as string;
|
||||
}
|
||||
|
||||
removeColors(e)
|
||||
.split("\n")
|
||||
.map((text, index, { length }) => {
|
||||
line.push({ text, formats });
|
||||
if (index < length - 1) {
|
||||
newLine();
|
||||
// removeColors(msg) // Remove colors is uncommented for now, since there are no real benefits of having it and it reduces performance
|
||||
msg.split("\n").forEach((text, index, { length }) => {
|
||||
if (index != length - 1) {
|
||||
formattedMessage.push(
|
||||
new Formatted(text + "\n", format),
|
||||
...linePrefix
|
||||
);
|
||||
} else {
|
||||
formattedMessage.push(new Formatted(text, format));
|
||||
}
|
||||
});
|
||||
|
||||
if (!e.endsWith("\n") && i < message.length - 1) {
|
||||
line.push({ text: " ", formats: [] });
|
||||
}
|
||||
formattedMessage.push(new Formatted(" "));
|
||||
});
|
||||
|
||||
newLine();
|
||||
|
||||
let msg: Message = {
|
||||
date: new Date(),
|
||||
date,
|
||||
file,
|
||||
name: this._name,
|
||||
text: {
|
||||
raw,
|
||||
formatted,
|
||||
},
|
||||
names,
|
||||
text: formattedMessage,
|
||||
type,
|
||||
};
|
||||
|
||||
this.messageObservable.send(msg);
|
||||
this.#adapters.forEach((adapter) => {
|
||||
if (adapter.level <= type) {
|
||||
adapter.onMessage(msg);
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
const colorSymbol = Symbol("color");
|
||||
async close() {
|
||||
if (this.#closed) return;
|
||||
this.#closed = true;
|
||||
|
||||
export interface ColorFormat {
|
||||
[colorSymbol]: Colors;
|
||||
value: any;
|
||||
}
|
||||
|
||||
export function withColor(color: Colors, value: any): ColorFormat {
|
||||
return {
|
||||
[colorSymbol]: color,
|
||||
value,
|
||||
};
|
||||
for (const adapter of this.#adapters) {
|
||||
const cnt = LoggingBase[InitialisedAdapters].get(adapter);
|
||||
if (!cnt) {
|
||||
//TODO: should not happen!
|
||||
} else {
|
||||
if (cnt <= 1) {
|
||||
if (adapter.close) await adapter.close();
|
||||
LoggingBase[InitialisedAdapters].delete(adapter);
|
||||
} else {
|
||||
LoggingBase[InitialisedAdapters].set(adapter, cnt - 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function getStack() {
|
||||
@ -503,13 +455,15 @@ function getCallerFile() {
|
||||
};
|
||||
}
|
||||
}
|
||||
} catch (err) {}
|
||||
} catch (err) { }
|
||||
return { file: undefined, line: 0 };
|
||||
}
|
||||
|
||||
function getCallerFromExisting(
|
||||
err: Error
|
||||
): { file: string; line: number; column?: number } {
|
||||
function getCallerFromExisting(err: Error): {
|
||||
file: string;
|
||||
line: number;
|
||||
column?: number;
|
||||
} {
|
||||
if (!err || !err.stack) return { file: "NOFILE", line: 0 };
|
||||
let lines = err.stack.split("\n");
|
||||
lines.shift(); // removing first line
|
||||
@ -537,3 +491,13 @@ function getCallerFromExisting(
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
export function removeColors(text: string) {
|
||||
text = text.replace(
|
||||
// Putting regex here directly instead of externally actually improves performance. The cause of that is not clear for now.
|
||||
/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g,
|
||||
""
|
||||
);
|
||||
|
||||
return text;
|
||||
}
|
||||
|
@ -1,26 +1,30 @@
|
||||
import { ObservableInterface } from "@hibas123/utils";
|
||||
import { Colors } from "./index.js";
|
||||
import {
|
||||
Adapter,
|
||||
Message,
|
||||
FormattedLine,
|
||||
TerminalFormats,
|
||||
FormatTypes,
|
||||
Formatted,
|
||||
IFormatted,
|
||||
LoggingTypes,
|
||||
} from "./types.js";
|
||||
|
||||
declare const Deno: any;
|
||||
declare const process: any;
|
||||
|
||||
const browser = typeof window !== "undefined" && typeof Deno === "undefined";
|
||||
const deno = typeof Deno !== "undefined";
|
||||
const NodeJS = typeof process !== undefined;
|
||||
|
||||
export class ConsoleAdapter implements Adapter {
|
||||
constructor(private colors: boolean = true) {}
|
||||
level: LoggingTypes = LoggingTypes.Debug;
|
||||
constructor(private colors: boolean = true) { }
|
||||
|
||||
init(observable: ObservableInterface<Message>) {
|
||||
observable.subscribe(this.onMessage.bind(this));
|
||||
init() { }
|
||||
flush() { }
|
||||
|
||||
setLevel(level: LoggingTypes) {
|
||||
this.level = level;
|
||||
}
|
||||
|
||||
flush() {}
|
||||
|
||||
// TODO: Check if required!
|
||||
// private escape(text: string): string {
|
||||
// return text
|
||||
@ -28,25 +32,29 @@ export class ConsoleAdapter implements Adapter {
|
||||
// .replace(/%c/g, "%%c")
|
||||
// }
|
||||
|
||||
private formatLine(line: FormattedLine): [string, string[] | undefined] {
|
||||
private format(
|
||||
formatted: IFormatted<string>[]
|
||||
): [string, string[] | undefined] {
|
||||
let text = "";
|
||||
let style_formats: string[] = [];
|
||||
|
||||
if (!browser) {
|
||||
for (let part of line) {
|
||||
// NodeJS or Deno
|
||||
for (const format of formatted) {
|
||||
let formats = "";
|
||||
for (let format of part.formats) {
|
||||
switch (format.type) {
|
||||
case FormatTypes.BOLD:
|
||||
if (format.bold) {
|
||||
formats += TerminalFormats.Bold;
|
||||
break;
|
||||
case FormatTypes.UNDERSCORE:
|
||||
formats += TerminalFormats.Underscore;
|
||||
break;
|
||||
case FormatTypes.BLINK:
|
||||
}
|
||||
|
||||
if (format.blink) {
|
||||
formats += TerminalFormats.Blink;
|
||||
break;
|
||||
case FormatTypes.COLOR:
|
||||
}
|
||||
|
||||
if (format.underscore) {
|
||||
formats += TerminalFormats.Underscore;
|
||||
}
|
||||
|
||||
if (format.color) {
|
||||
switch (format.color) {
|
||||
case Colors.RED:
|
||||
formats += TerminalFormats.FgRed;
|
||||
@ -70,49 +78,76 @@ export class ConsoleAdapter implements Adapter {
|
||||
formats += TerminalFormats.FgWhite;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
if (format.bgcolor) {
|
||||
switch (format.bgcolor) {
|
||||
case Colors.RED:
|
||||
formats += TerminalFormats.BgRed;
|
||||
break;
|
||||
case Colors.GREEN:
|
||||
formats += TerminalFormats.BgGreen;
|
||||
break;
|
||||
case Colors.YELLOW:
|
||||
formats += TerminalFormats.BgYellow;
|
||||
break;
|
||||
case Colors.BLUE:
|
||||
formats += TerminalFormats.BgBlue;
|
||||
break;
|
||||
case Colors.MAGENTA:
|
||||
formats += TerminalFormats.BgMagenta;
|
||||
break;
|
||||
case Colors.CYAN:
|
||||
formats += TerminalFormats.BgCyan;
|
||||
break;
|
||||
case Colors.WHITE:
|
||||
formats += TerminalFormats.BgWhite;
|
||||
break;
|
||||
}
|
||||
}
|
||||
text += formats + part.text + TerminalFormats.Reset;
|
||||
|
||||
text += formats + format.content + TerminalFormats.Reset;
|
||||
// (formats.length > 0 ? TerminalFormats.Reset : ""); //TODO: Benchmark if this is better
|
||||
}
|
||||
} else {
|
||||
for (let part of line) {
|
||||
for (const format of formatted) {
|
||||
let styles: string[] = [];
|
||||
let resetStyles: string[] = [];
|
||||
for (let format of part.formats) {
|
||||
switch (format.type) {
|
||||
case FormatTypes.BOLD:
|
||||
if (format.bold) {
|
||||
styles.push("font-weight: bold;");
|
||||
resetStyles.push("font-weight: unset");
|
||||
break;
|
||||
case FormatTypes.UNDERSCORE:
|
||||
}
|
||||
|
||||
if (format.underscore) {
|
||||
styles.push("text-decoration: underline");
|
||||
resetStyles.push("text-decoration: unset");
|
||||
break;
|
||||
case FormatTypes.BLINK:
|
||||
}
|
||||
|
||||
if (format.blink) {
|
||||
styles.push("text-decoration: blink");
|
||||
resetStyles.push("text-decoration: unset");
|
||||
break;
|
||||
case FormatTypes.COLOR:
|
||||
}
|
||||
|
||||
if (format.color) {
|
||||
let color = "";
|
||||
switch (format.color) {
|
||||
case Colors.RED:
|
||||
color = "red";
|
||||
color = "#ff5f5f";
|
||||
break;
|
||||
case Colors.GREEN:
|
||||
color = "green";
|
||||
color = "#62ff5f";
|
||||
break;
|
||||
case Colors.YELLOW:
|
||||
color = "gold";
|
||||
color = "#ffea37";
|
||||
break;
|
||||
case Colors.BLUE:
|
||||
color = "blue";
|
||||
color = "#379aff";
|
||||
break;
|
||||
case Colors.MAGENTA:
|
||||
color = "magenta";
|
||||
color = "#f837ff";
|
||||
break;
|
||||
case Colors.CYAN:
|
||||
color = "cyan";
|
||||
color = "#37e4ff";
|
||||
break;
|
||||
case Colors.WHITE:
|
||||
color = "white";
|
||||
@ -120,10 +155,38 @@ export class ConsoleAdapter implements Adapter {
|
||||
}
|
||||
styles.push("color: " + color);
|
||||
resetStyles.push("color: unset");
|
||||
}
|
||||
|
||||
if (format.bgcolor) {
|
||||
let color = "";
|
||||
switch (format.bgcolor) {
|
||||
case Colors.RED:
|
||||
color = "#ff5f5f";
|
||||
break;
|
||||
case Colors.GREEN:
|
||||
color = "#62ff5f";
|
||||
break;
|
||||
case Colors.YELLOW:
|
||||
color = "#ffea37";
|
||||
break;
|
||||
case Colors.BLUE:
|
||||
color = "#379aff";
|
||||
break;
|
||||
case Colors.MAGENTA:
|
||||
color = "#f837ff";
|
||||
break;
|
||||
case Colors.CYAN:
|
||||
color = "#37e4ff";
|
||||
break;
|
||||
case Colors.WHITE:
|
||||
color = "white";
|
||||
break;
|
||||
}
|
||||
styles.push("background-color: " + color);
|
||||
resetStyles.push("background-color: unset");
|
||||
}
|
||||
text += "%c" + part.text.replace(/%c/g, "%%c") + "%c";
|
||||
|
||||
text += "%c" + format.content.replace(/%c/g, "%%c") + "%c";
|
||||
style_formats.push(styles.join(";"), resetStyles.join(";"));
|
||||
}
|
||||
}
|
||||
@ -132,34 +195,26 @@ export class ConsoleAdapter implements Adapter {
|
||||
}
|
||||
|
||||
onMessage(message: Message) {
|
||||
let lines = message.text.formatted;
|
||||
|
||||
let prefix = "";
|
||||
if (message.name) prefix = `[${message.name}]=>`;
|
||||
|
||||
if (browser) {
|
||||
if (this.colors) {
|
||||
let formats: string[] = [];
|
||||
let text = lines
|
||||
.map((line) => {
|
||||
let [t, fmts] = this.formatLine(line);
|
||||
formats.push(...fmts);
|
||||
return prefix + t;
|
||||
})
|
||||
.join("\n");
|
||||
// console.log(formats);
|
||||
const [text, formats] = this.format(message.text);
|
||||
console.log(text, ...formats);
|
||||
} else {
|
||||
console.log(message.text.raw.join("\n"));
|
||||
console.log(Formatted.strip(message.text));
|
||||
}
|
||||
} else {
|
||||
if (this.colors) {
|
||||
lines.forEach((line) => {
|
||||
let [text] = this.formatLine(line);
|
||||
console.log(prefix + text);
|
||||
});
|
||||
const text = this.colors
|
||||
? this.format(message.text)[0]
|
||||
: Formatted.strip(message.text);
|
||||
|
||||
if (deno) {
|
||||
//TODO: Deno specific thing
|
||||
console.log(text);
|
||||
} else if (typeof process !== "undefined") {
|
||||
//NodeJS
|
||||
process.stdout.write(text + "\n");
|
||||
} else {
|
||||
message.text.raw.forEach((line) => console.log(prefix + line));
|
||||
console.log(text);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
22
src/index.ts
22
src/index.ts
@ -1,27 +1,23 @@
|
||||
import { LoggingBase } from "./base.js";
|
||||
export { LoggingBase } from "./base.js";
|
||||
export { ConsoleAdapter } from "./consolewriter.js";
|
||||
export {
|
||||
LoggingBase,
|
||||
LoggingBaseOptions,
|
||||
removeColors,
|
||||
withColor,
|
||||
} from "./base.js";
|
||||
export { ILoggingOptions, INativeFunctions } from "./base.js";
|
||||
|
||||
export {
|
||||
Adapter,
|
||||
LoggingTypes,
|
||||
Message,
|
||||
FormatConfig,
|
||||
FormattedLine,
|
||||
DefaultFormatConfig as DefaultColorMap,
|
||||
FormattedText,
|
||||
Colors,
|
||||
IColors as Colors,
|
||||
Format,
|
||||
FormatTypes,
|
||||
TerminalFormats,
|
||||
Formatted,
|
||||
IFormatted,
|
||||
ILoggingInterface,
|
||||
ILoggingTimer,
|
||||
} from "./types.js";
|
||||
|
||||
export { ObservableInterface } from "@hibas123/utils";
|
||||
const Logging = new LoggingBase();
|
||||
|
||||
export let Logging: LoggingBase = undefined;
|
||||
Logging = new LoggingBase();
|
||||
export default Logging;
|
||||
|
49
src/test.ts
49
src/test.ts
@ -1,26 +1,19 @@
|
||||
import {
|
||||
Logging,
|
||||
LoggingBase,
|
||||
LoggingTypes,
|
||||
Colors,
|
||||
withColor,
|
||||
} from "./index.js";
|
||||
import Logging from "./index.js";
|
||||
|
||||
import { LoggingBase, LoggingTypes, Format } from "./index.js";
|
||||
|
||||
Logging.log("test");
|
||||
Logging.log("i", "am", { a: "an" }, 1000);
|
||||
Logging.error(new Error("fehler 001"));
|
||||
Logging.debug("Some Debug infos");
|
||||
Logging.errorMessage("i", "am", "an", "error");
|
||||
Logging.error("i", "am", "an", "error");
|
||||
|
||||
Logging.log(
|
||||
"\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m TEST \x1b[31m\x1b[31m\x1b[31m"
|
||||
);
|
||||
|
||||
Logging.log(
|
||||
withColor(Colors.MAGENTA, "This text should be magenta!"),
|
||||
"This not!"
|
||||
);
|
||||
Logging.log(withColor(Colors.MAGENTA, { somekey: "Some value" }));
|
||||
Logging.log(Format.magenta("This text should be magenta!"), "This not!");
|
||||
Logging.log(Format.magenta({ somekey: "Some value" }));
|
||||
|
||||
let err = new Error();
|
||||
if (typeof err.stack !== "string") console.log("Stacktrace invalid", err.stack);
|
||||
@ -29,10 +22,10 @@ let cus = new LoggingBase({ name: "test" });
|
||||
cus.log("Hello from custom Logger");
|
||||
cus.log("This has some %c symbols inside of it!");
|
||||
|
||||
let cus2 = new LoggingBase("test2");
|
||||
let cus2 = new LoggingBase({ name: "test2" });
|
||||
cus2.log("Hello from custom Logger 2");
|
||||
|
||||
let cus22 = new LoggingBase("test2");
|
||||
let cus22 = new LoggingBase({ name: "test2" });
|
||||
cus22.log("Hello from custom Logger 22");
|
||||
cus2.log("Hello from custom Logger 2");
|
||||
cus22.log("Hello from custom Logger 22");
|
||||
@ -53,7 +46,9 @@ Logging.warn("This should not be there 3");
|
||||
Logging.warning("This should not be there 4");
|
||||
|
||||
Logging.error("This should be there 1");
|
||||
Logging.errorMessage("This should be there 2");
|
||||
Logging.error("This should be there 2");
|
||||
|
||||
Logging.logLevel = LoggingTypes.Debug;
|
||||
|
||||
const c1 = Logging.getChild("child-level-1");
|
||||
|
||||
@ -63,23 +58,15 @@ const c2 = c1.getChild("child-level-2");
|
||||
|
||||
c2.log("Hello from Child 2");
|
||||
|
||||
c2.addAdapter({
|
||||
init: (obs) =>
|
||||
obs.subscribe((msg) =>
|
||||
console.log(
|
||||
"Adapter adden on child level 2: ",
|
||||
"---",
|
||||
msg.name,
|
||||
msg.text.raw
|
||||
)
|
||||
),
|
||||
flush: () => undefined,
|
||||
close: () => undefined,
|
||||
});
|
||||
|
||||
c2.log("MSG from C2");
|
||||
c1.log("MSG from C1");
|
||||
Logging.log("MSG from root");
|
||||
|
||||
const timer = Logging.time("timer1", "Test Timer");
|
||||
const timer = Logging.time("timer1");
|
||||
setTimeout(() => timer.end(), 1000);
|
||||
|
||||
const withoutFile = new LoggingBase({
|
||||
resolve_filename: false,
|
||||
});
|
||||
|
||||
withoutFile.log("This should not have a file attached");
|
||||
|
230
src/types.ts
230
src/types.ts
@ -1,9 +1,7 @@
|
||||
import { ObservableInterface } from "@hibas123/utils";
|
||||
|
||||
export enum LoggingTypes {
|
||||
Debug,
|
||||
Log,
|
||||
Warning,
|
||||
Warn,
|
||||
Error,
|
||||
}
|
||||
|
||||
@ -34,14 +32,7 @@ export const TerminalFormats = {
|
||||
BgWhite: "\x1b[47m",
|
||||
};
|
||||
|
||||
export enum FormatTypes {
|
||||
COLOR,
|
||||
BOLD,
|
||||
UNDERSCORE,
|
||||
BLINK,
|
||||
}
|
||||
|
||||
export enum Colors {
|
||||
export enum IColors {
|
||||
NONE,
|
||||
RED,
|
||||
GREEN,
|
||||
@ -52,70 +43,194 @@ export enum Colors {
|
||||
WHITE,
|
||||
}
|
||||
|
||||
export interface FormatConfig {
|
||||
error: Format[];
|
||||
warning: Format[];
|
||||
log: Format[];
|
||||
debug: Format[];
|
||||
export interface IFormatted<T = any> {
|
||||
color?: IColors;
|
||||
bgcolor?: IColors;
|
||||
bold?: boolean;
|
||||
italic?: boolean;
|
||||
blink?: boolean;
|
||||
underscore?: boolean;
|
||||
|
||||
date: Format[];
|
||||
file: Format[];
|
||||
content: T;
|
||||
}
|
||||
|
||||
function colorFormat(color: Colors) {
|
||||
return {
|
||||
type: FormatTypes.COLOR,
|
||||
color,
|
||||
};
|
||||
export class Formatted<T = any> implements IFormatted<T> {
|
||||
static strip(formatted: IFormatted<string> | IFormatted<string>[]) {
|
||||
if (Array.isArray(formatted)) {
|
||||
return formatted.reduce((p, c) => p + c.content, "");
|
||||
} else {
|
||||
return formatted.content;
|
||||
}
|
||||
}
|
||||
|
||||
constructor(content?: T, apply?: Formatted) {
|
||||
if (content instanceof Formatted) {
|
||||
this.color = content.color;
|
||||
this.bgcolor = content.bgcolor;
|
||||
this.bold = content.bold;
|
||||
this.italic = content.italic;
|
||||
this.blink = content.blink;
|
||||
this.underscore = content.underscore;
|
||||
this.content = content.content;
|
||||
} else {
|
||||
this.content = content;
|
||||
}
|
||||
|
||||
if (apply) {
|
||||
this.color = apply.color;
|
||||
this.bgcolor = apply.bgcolor;
|
||||
this.bold = apply.bold;
|
||||
this.italic = apply.italic;
|
||||
this.blink = apply.blink;
|
||||
this.underscore = apply.underscore;
|
||||
}
|
||||
}
|
||||
|
||||
color?: IColors;
|
||||
bgcolor?: IColors;
|
||||
bold?: boolean;
|
||||
italic?: boolean;
|
||||
blink?: boolean;
|
||||
underscore?: boolean;
|
||||
|
||||
content: T;
|
||||
|
||||
_color(c: IColors) {
|
||||
this.color = c;
|
||||
return this;
|
||||
}
|
||||
|
||||
_bgcolor(c: IColors) {
|
||||
this.bgcolor = c;
|
||||
return this;
|
||||
}
|
||||
|
||||
_bold(active = true) {
|
||||
this.bold = active;
|
||||
return this;
|
||||
}
|
||||
|
||||
_italic(active = true) {
|
||||
this.italic = active;
|
||||
return this;
|
||||
}
|
||||
|
||||
_blink(active = true) {
|
||||
this.blink = active;
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
const boldFormat = {
|
||||
type: FormatTypes.BOLD,
|
||||
const AddColorFormat = (color: IColors) => (content: any) => {
|
||||
if (content instanceof Formatted) {
|
||||
content.color = color;
|
||||
return content;
|
||||
} else {
|
||||
const d = new Formatted(content);
|
||||
d.color = color;
|
||||
return d;
|
||||
}
|
||||
};
|
||||
|
||||
const AddBGColorFormat = (color: IColors) => (content: any) => {
|
||||
if (content instanceof Formatted) {
|
||||
content.bgcolor = color;
|
||||
return content;
|
||||
} else {
|
||||
const d = new Formatted(content);
|
||||
d.bgcolor = color;
|
||||
return d;
|
||||
}
|
||||
};
|
||||
|
||||
export const Format = {
|
||||
none: AddColorFormat(IColors.NONE),
|
||||
red: AddColorFormat(IColors.RED),
|
||||
green: AddColorFormat(IColors.GREEN),
|
||||
yellow: AddColorFormat(IColors.YELLOW),
|
||||
blue: AddColorFormat(IColors.BLUE),
|
||||
magenta: AddColorFormat(IColors.MAGENTA),
|
||||
cyan: AddColorFormat(IColors.CYAN),
|
||||
white: AddColorFormat(IColors.WHITE),
|
||||
bgnone: AddBGColorFormat(IColors.NONE),
|
||||
bgred: AddBGColorFormat(IColors.RED),
|
||||
bggreen: AddBGColorFormat(IColors.GREEN),
|
||||
bgyellow: AddBGColorFormat(IColors.YELLOW),
|
||||
bgblue: AddBGColorFormat(IColors.BLUE),
|
||||
bgmagenta: AddBGColorFormat(IColors.MAGENTA),
|
||||
bgcyan: AddBGColorFormat(IColors.CYAN),
|
||||
bgwhite: AddBGColorFormat(IColors.WHITE),
|
||||
bold: (content: any) => {
|
||||
if (content instanceof Formatted) {
|
||||
content.bold = true;
|
||||
return content;
|
||||
} else {
|
||||
const d = new Formatted(content);
|
||||
d.bold = true;
|
||||
return d;
|
||||
}
|
||||
},
|
||||
italic: (content: any) => {
|
||||
if (content instanceof Formatted) {
|
||||
content.italic = true;
|
||||
return content;
|
||||
} else {
|
||||
const d = new Formatted(content);
|
||||
d.italic = true;
|
||||
return d;
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
export interface FormatConfig {
|
||||
error: Formatted;
|
||||
warning: Formatted;
|
||||
log: Formatted;
|
||||
debug: Formatted;
|
||||
|
||||
date: Formatted;
|
||||
file: Formatted;
|
||||
|
||||
names: Formatted;
|
||||
names_delimiter: Formatted;
|
||||
}
|
||||
|
||||
export class DefaultFormatConfig implements FormatConfig {
|
||||
error = [colorFormat(Colors.RED), boldFormat];
|
||||
warning = [colorFormat(Colors.YELLOW), boldFormat];
|
||||
log = [colorFormat(Colors.NONE), boldFormat];
|
||||
debug = [colorFormat(Colors.CYAN), boldFormat];
|
||||
error = new Formatted()._color(IColors.RED)._bold();
|
||||
warning = new Formatted()._color(IColors.YELLOW)._bold();
|
||||
log = new Formatted()._color(IColors.NONE)._bold();
|
||||
debug = new Formatted()._color(IColors.CYAN)._bold();
|
||||
|
||||
date = [colorFormat(Colors.NONE)];
|
||||
file = [colorFormat(Colors.NONE)];
|
||||
date = new Formatted()._color(IColors.NONE);
|
||||
file = new Formatted()._color(IColors.NONE);
|
||||
names = new Formatted()._bold()._color(IColors.CYAN);
|
||||
names_delimiter = new Formatted(" -> ")._bold();
|
||||
}
|
||||
|
||||
export interface Format {
|
||||
type: FormatTypes;
|
||||
color?: Colors;
|
||||
}
|
||||
|
||||
export interface FormattedText {
|
||||
text: string;
|
||||
formats: Format[];
|
||||
}
|
||||
|
||||
export type FormattedLine = FormattedText[];
|
||||
|
||||
export interface Message {
|
||||
type: LoggingTypes;
|
||||
name?: string;
|
||||
text: {
|
||||
raw: string[];
|
||||
formatted: FormattedLine[];
|
||||
};
|
||||
names?: string[];
|
||||
text: IFormatted<string>[];
|
||||
date: Date;
|
||||
file: string;
|
||||
}
|
||||
|
||||
export interface Adapter {
|
||||
readonly level: LoggingTypes;
|
||||
|
||||
/**
|
||||
* This function initialises the Adapter. It might be called multiple times, when added to multiple instances
|
||||
* @param observable An observable to subscribe to messages
|
||||
*/
|
||||
init(observable: ObservableInterface<Message>): void | Promise<void>;
|
||||
init(): void | Promise<void>;
|
||||
|
||||
flush(sync: true): void;
|
||||
flush(sync: false): void | Promise<void>;
|
||||
|
||||
setLevel(level: LoggingTypes): void;
|
||||
|
||||
onMessage(message: Message): void;
|
||||
|
||||
/**
|
||||
* When a close function is available, it will be called when no logging instance references it anymore.
|
||||
*
|
||||
@ -123,3 +238,20 @@ export interface Adapter {
|
||||
*/
|
||||
close?(): void;
|
||||
}
|
||||
|
||||
export interface ILoggingTimer {
|
||||
mark: (label: string) => void;
|
||||
end: () => number;
|
||||
}
|
||||
|
||||
export interface ILoggingInterface {
|
||||
debug(...message: any[]): void;
|
||||
log(...message: any[]): void;
|
||||
warn(...message: any[]): void;
|
||||
error(...message: any[]): void;
|
||||
|
||||
time(name?: string): ILoggingTimer;
|
||||
timeEnd(id: string): number;
|
||||
|
||||
getChild(name: string): ILoggingInterface;
|
||||
}
|
||||
|
@ -1,11 +0,0 @@
|
||||
{
|
||||
"extends": "./tsconfig.json",
|
||||
"compilerOptions": {
|
||||
"module": "ESNext",
|
||||
"target": "ES2017",
|
||||
"moduleResolution": "node",
|
||||
"outDir": "esm"
|
||||
},
|
||||
"exclude": ["node_modules"],
|
||||
"include": ["src"]
|
||||
}
|
@ -1,12 +1,12 @@
|
||||
{
|
||||
"compilerOptions": {
|
||||
"module": "commonjs",
|
||||
"target": "esnext",
|
||||
"module": "ESNext",
|
||||
"target": "ESNext",
|
||||
"moduleResolution": "node",
|
||||
"outDir": "esm",
|
||||
"noImplicitAny": false,
|
||||
"sourceMap": true,
|
||||
"outDir": "out",
|
||||
"declaration": true,
|
||||
"typeRoots": ["node_modules/@types"]
|
||||
"declaration": true
|
||||
},
|
||||
"exclude": ["node_modules"],
|
||||
"include": ["src"]
|
||||
|
Reference in New Issue
Block a user