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", |          "request": "launch", | ||||||
|          "name": "Launch Program", |          "name": "Launch Program", | ||||||
|          "skipFiles": ["<node_internals>/**"], |          "skipFiles": ["<node_internals>/**"], | ||||||
|          "program": "${workspaceFolder}\\out\\test.js", |          "program": "${workspaceFolder}/out/test.js", | ||||||
|          "preLaunchTask": "tsc: build - tsconfig.json", |          "preLaunchTask": "tsc: build - tsconfig.json", | ||||||
|          "outFiles": ["${workspaceFolder}/out/**/*.js"] |          "outFiles": ["${workspaceFolder}/out/**/*.js"] | ||||||
|       } |       } | ||||||
|  | |||||||
| @ -1,29 +1,29 @@ | |||||||
|  | # Logging | ||||||
|  | 
 | ||||||
| Simple logging module, that supports terminal coloring and different plugins | Simple logging module, that supports terminal coloring and different plugins | ||||||
| 
 | 
 | ||||||
| # Getting Started | ## Getting Started | ||||||
| 
 | 
 | ||||||
| ```javascript | ```javascript | ||||||
| 
 | const Logging = require("@hibas123/logging").default; | ||||||
| const Logging = require("@hibas123/logging").Logging; |  | ||||||
| 
 | 
 | ||||||
| Logging.log("Hello there"); | Logging.log("Hello there"); | ||||||
| 
 |  | ||||||
| ``` | ``` | ||||||
| 
 | 
 | ||||||
| There are different Logging levels, that also apply terminal coloring: | There are different Logging levels, that also apply terminal coloring: | ||||||
| 
 | 
 | ||||||
| ```javascript | ```javascript | ||||||
| Logging.debug("Debug message") | Logging.debug("Debug message"); | ||||||
| Logging.log("Log message") | Logging.log("Log message"); | ||||||
| Logging.warning("Warning") | Logging.warning("Warning"); | ||||||
| Logging.error(new Error("To less creativity")) | Logging.error(new Error("To less creativity")); | ||||||
| Logging.error("Just an simple message as error") | Logging.error("Just an simple message as error"); | ||||||
| Logging.errorMessage("Nearly the same 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. | 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. | This logging module doesn't require any setup per default, but sometimes it makes sense to configure a few things. | ||||||
| 
 | 
 | ||||||
| @ -32,14 +32,16 @@ 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. | Also you can set a name. All messages that are send with this instance are prefixed by this name. | ||||||
| 
 | 
 | ||||||
| ```javascript | ```javascript | ||||||
|    const CustomLogging = new LoggingBase(name | { | const CustomLogging = new LoggingBase( | ||||||
|  |    name | | ||||||
|  |       { | ||||||
|          name: "custom", // default undefined |          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. | There is a Plugin API available, that makes is possible to add custom Logging Adapter. | ||||||
| 
 | 
 | ||||||
| @ -52,7 +54,9 @@ The adapters need to provide a very simple Interface: | |||||||
| 
 | 
 | ||||||
| ```typescript | ```typescript | ||||||
| interface Adapter { | 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: true): void; | ||||||
|    flush(sync: false): void | Promise<void>; |    flush(sync: false): void | Promise<void>; | ||||||
| @ -60,28 +64,33 @@ interface Adapter { | |||||||
| 
 | 
 | ||||||
| interface Message { | interface Message { | ||||||
|    type: LoggingTypes; |    type: LoggingTypes; | ||||||
|     name?:string; |    names?: string[]; | ||||||
|     text: { |    text: IFormatted; | ||||||
|     	raw: string[], |  | ||||||
|     	formatted: string[] |  | ||||||
|     }; |  | ||||||
|    date: Date; |    date: Date; | ||||||
|    file: string; |    file: string; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | export interface IFormatted { | ||||||
|  |    color?: IColors; | ||||||
|  |    bgcolor?: IColors; | ||||||
|  |    bold?: boolean; | ||||||
|  |    italic?: boolean; | ||||||
|  |    blink?: boolean; | ||||||
|  |    underscore?: boolean; | ||||||
|  | 
 | ||||||
|  |    content: string; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| enum LoggingTypes { | enum LoggingTypes { | ||||||
|    Log, |    Log, | ||||||
|    Warning, |    Warning, | ||||||
|    Error, |    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 | MIT | ||||||
| 
 | 
 | ||||||
| Copyright (c) 2018 Fabian Stamm | 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", |    "name": "@hibas123/logging", | ||||||
|    "version": "2.5.4", |    "version": "4.0.2", | ||||||
|    "description": "", |    "description": "", | ||||||
|    "main": "out/index.js", |    "type": "module", | ||||||
|    "types": "out/index.d.ts", |    "main": "esm/index.js", | ||||||
|  |    "types": "esm/index.d.ts", | ||||||
|    "module": "esm/index.js", |    "module": "esm/index.js", | ||||||
|    "scripts": { |    "scripts": { | ||||||
|       "prepublish": "npm run build", |       "prepublishOnly": "npm run build", | ||||||
|       "build": "tsc && tsc -p tsconfig.esm.json", |       "build": "tsc", | ||||||
|       "dev": "nodemon -e ts --exec ts-node src/test.ts" |       "test": "tsc && node esm/test.js", | ||||||
|  |       "bench": "tsc && node benchmark.js", | ||||||
|  |       "prof": "tsc && ndb --prof benchmark.js" | ||||||
|    }, |    }, | ||||||
|    "repository": { |    "repository": { | ||||||
|       "type": "git", |       "type": "git", | ||||||
| @ -18,18 +21,15 @@ | |||||||
|    "license": "MIT", |    "license": "MIT", | ||||||
|    "files": [ |    "files": [ | ||||||
|       "src/", |       "src/", | ||||||
|       "out/", |  | ||||||
|       "esm/", |       "esm/", | ||||||
|       "tsconfig.json", |       "tsconfig.json", | ||||||
|       "readme.md" |       "readme.md" | ||||||
|    ], |    ], | ||||||
|    "devDependencies": { |    "devDependencies": { | ||||||
|       "concurrently": "^5.3.0", |       "concurrently": "^9.2.1", | ||||||
|       "nodemon": "^2.0.4", |       "ndb": "^1.1.5", | ||||||
|       "ts-node": "^9.0.0", |       "nodemon": "^3.1.10", | ||||||
|       "typescript": "^4.0.2" |       "ts-node": "^10.9.2", | ||||||
|    }, |       "typescript": "^5.9.3" | ||||||
|    "dependencies": { |  | ||||||
|       "@hibas123/utils": "^2.2.10" |  | ||||||
|    } |    } | ||||||
| } | } | ||||||
							
								
								
									
										658
									
								
								src/base.ts
									
									
									
									
									
								
							
							
						
						
									
										658
									
								
								src/base.ts
									
									
									
									
									
								
							| @ -1,36 +1,19 @@ | |||||||
| import { Observable, ObservableInterface } from "@hibas123/utils"; |  | ||||||
| import { ConsoleAdapter } from "./consolewriter.js"; | import { ConsoleAdapter } from "./consolewriter.js"; | ||||||
| import inspect from "./inspect.js"; | import inspect from "./inspect.js"; | ||||||
| import { | import { | ||||||
|    Adapter, |    Adapter, | ||||||
|    LoggingTypes, |  | ||||||
|    Message, |  | ||||||
|    FormatConfig, |  | ||||||
|    DefaultFormatConfig, |    DefaultFormatConfig, | ||||||
|    Format, |    Format, | ||||||
|    FormatTypes, |    FormatConfig, | ||||||
|    Colors, |    Formatted, | ||||||
|    FormattedText, |    ILoggingInterface, | ||||||
|    FormattedLine, |    LoggingTypes, | ||||||
|  |    Message, | ||||||
| } from "./types.js"; | } from "./types.js"; | ||||||
|  |  | ||||||
| const browser = typeof window !== "undefined"; | const browser = typeof window !== "undefined"; | ||||||
|  |  | ||||||
| export function removeColors(text: string) { | export interface ILoggingOptions { | ||||||
|    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 { |  | ||||||
|    /** |    /** | ||||||
|     * Name will be prefixed on Console output and added to logfiles, if not specified here |     * 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 |     * Prints output to console | ||||||
|     */ |     */ | ||||||
|    console: boolean; |    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 |     * Enables printing of calling file | ||||||
|     * @param child New child logging instance |  | ||||||
|     */ |     */ | ||||||
|    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; |  | ||||||
| } | } | ||||||
|  |  | ||||||
|    private initAdapter(adapter: Adapter) { | export interface INativeFunctions { | ||||||
|       let cached = adapterCache.get(adapter) || 0; |    startTimer(): any; | ||||||
|       adapterCache.set(adapter, cached + 1); |    diffTime(start: any, end: any): number; | ||||||
|  |    endTimer(start: any): number; | ||||||
|       let prms = Promise.resolve( |  | ||||||
|          adapter.init(this.messageObservable.getPublicApi()) |  | ||||||
|       ); |  | ||||||
|       this.adapter_init.push(prms); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|    addAdapter(adapter: Adapter) { | export const DefaultNativeFunctions = { | ||||||
|       this.adapterSet.addAdapter(adapter); |    startTimer: () => { | ||||||
|    } |  | ||||||
|  |  | ||||||
|    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 { |  | ||||||
|       if (browser && window.performance && window.performance.now) { |       if (browser && window.performance && window.performance.now) { | ||||||
|          return window.performance.now(); |          return window.performance.now(); | ||||||
|       } else { |       } else { | ||||||
|          return Date.now(); |          return Date.now(); | ||||||
|       } |       } | ||||||
|    } |    }, | ||||||
|  |    diffTime: (start: any, end: any) => { | ||||||
|    /** |       return end - start; | ||||||
|     * The time difference in milliseconds (fractions allowed!) |    }, | ||||||
|     * @param start Start time from getCurrentTime |    endTimer: (start: any) => { | ||||||
|     */ |  | ||||||
|    protected getTimeDiff(start: any) { |  | ||||||
|       if (browser && window.performance && window.performance.now) { |       if (browser && window.performance && window.performance.now) { | ||||||
|          return window.performance.now() - start; |          return window.performance.now() - start; | ||||||
|       } else { |       } else { | ||||||
|          return Date.now() - start; |          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) { |    protected abstract message( | ||||||
|       if (!id) { |       type: LoggingTypes, | ||||||
|          id = Math.floor(Math.random() * 899999 + 100000).toString(); |       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, { |    debug(...message: any[]) { | ||||||
|          name, |       this.message(LoggingTypes.Debug, this.#names, message); | ||||||
|          start: this.getCurrentTime(), |    } | ||||||
|       }); |  | ||||||
|  |    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 { |       return { | ||||||
|          id, |          id, | ||||||
|  |          mark: (label: string) => { | ||||||
|  |             timer.marks.push({ time: LoggingBase.nativeFunctions.startTimer(), name: label }); | ||||||
|  |          }, | ||||||
|          end: () => this.timeEnd(id), |          end: () => this.timeEnd(id), | ||||||
|       }; |       }; | ||||||
|    } |    } | ||||||
|  |  | ||||||
|    timeEnd(id: string) { |    timeEnd(id: string) { | ||||||
|       let timer = this.timerMap.get(id); |       let timer = this.#timerMap.get(id); | ||||||
|       if (timer) { |       if (timer) { | ||||||
|          let diff = this.getTimeDiff(timer.start); |          let end_time = LoggingBase.nativeFunctions.startTimer(); | ||||||
|          this.message(LoggingTypes.Debug, [ |          let diff = LoggingBase.nativeFunctions.diffTime(timer.start, end_time); | ||||||
|             withColor(Colors.GREEN, `[${timer.name}]`), |  | ||||||
|  |          this.message(LoggingTypes.Debug, this.#names, [ | ||||||
|  |             Format.green(`[${timer.name}]`), | ||||||
|             `->`, |             `->`, | ||||||
|             withColor(Colors.BLUE, diff.toFixed(4)), |             Format.blue(diff.toFixed(4)), | ||||||
|             "ms", |             "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 diff; | ||||||
|       } |       } | ||||||
|       return -1; |       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, |       type: LoggingTypes, | ||||||
|  |       names: string[], | ||||||
|       message: any[], |       message: any[], | ||||||
|       caller?: { file: string; line: number; column?: number } |       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; |          let file_raw = caller; | ||||||
|          if (!file_raw) { |          if (!file_raw) { | ||||||
|             try { |             try { | ||||||
| @ -348,121 +304,117 @@ export class LoggingBase { | |||||||
|  |  | ||||||
|             file_raw.file = newF; |             file_raw.file = newF; | ||||||
|          } |          } | ||||||
|       let file = `${file_raw.file || "<unknown>"}:${file_raw.line}:${ |          file = `${file_raw.file || "<unknown>"}:${file_raw.line}:${file_raw.column || 0 | ||||||
|          file_raw.column || 0 |  | ||||||
|             }`; |             }`; | ||||||
|  |       } | ||||||
|  |  | ||||||
|       let type_str = LoggingTypes[type].toUpperCase().padEnd(5, " "); |       let type_str = LoggingTypes[type].toUpperCase().padEnd(5, " "); | ||||||
|       let type_format: Format[] = []; |       let type_format: Formatted; | ||||||
|       switch (type) { |       switch (type) { | ||||||
|          case LoggingTypes.Log: |          case LoggingTypes.Log: | ||||||
|             type_format = this._formatMap.log; |             type_format = this.#format_map.log; | ||||||
|             break; |             break; | ||||||
|          case LoggingTypes.Error: |          case LoggingTypes.Error: | ||||||
|             type_format = this._formatMap.error; |             type_format = this.#format_map.error; | ||||||
|             break; |             break; | ||||||
|          case LoggingTypes.Debug: |          case LoggingTypes.Debug: | ||||||
|             type_format = this._formatMap.debug; |             type_format = this.#format_map.debug; | ||||||
|             break; |             break; | ||||||
|          case LoggingTypes.Warning: |          case LoggingTypes.Warn: | ||||||
|             type_format = this._formatMap.warning; |             type_format = this.#format_map.warning; | ||||||
|             break; |             break; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|       const prefix: FormattedText[] = []; |       const nameFormatted: Formatted<string>[] = []; | ||||||
|       const a = (text: string, formats: Format[] = []) => { |       if (names.length > 0) { | ||||||
|          prefix.push({ |          nameFormatted.push(new Formatted("[")); | ||||||
|             text, |          for (let i = 0; i < names.length; i++) { | ||||||
|             formats, |             nameFormatted.push(new Formatted(names[i], this.#format_map.names)); | ||||||
|          }); |             if (i < names.length - 1) { | ||||||
|       }; |                nameFormatted.push(this.#format_map.names_delimiter); | ||||||
|  |  | ||||||
|       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; |  | ||||||
|             } |             } | ||||||
|          } |          } | ||||||
|             if (typeof e !== "string") |          nameFormatted.push(new Formatted("]")); | ||||||
|                e = inspect(e, { |       } | ||||||
|                   colors: true, |  | ||||||
|  |       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, |                showHidden: true, | ||||||
|                depth: 3, |                depth: 3, | ||||||
|             }) as string; |             }) as string; | ||||||
|          } |          } | ||||||
|  |  | ||||||
|          removeColors(e) |          // removeColors(msg) // Remove colors is uncommented for now, since there are no real benefits of having it and it reduces performance | ||||||
|             .split("\n") |          msg.split("\n").forEach((text, index, { length }) => { | ||||||
|             .map((text, index, { length }) => { |             if (index != length - 1) { | ||||||
|                line.push({ text, formats }); |                formattedMessage.push( | ||||||
|                if (index < length - 1) { |                   new Formatted(text + "\n", format), | ||||||
|                   newLine(); |                   ...linePrefix | ||||||
|  |                ); | ||||||
|  |             } else { | ||||||
|  |                formattedMessage.push(new Formatted(text, format)); | ||||||
|             } |             } | ||||||
|          }); |          }); | ||||||
|  |          formattedMessage.push(new Formatted(" ")); | ||||||
|          if (!e.endsWith("\n") && i < message.length - 1) { |  | ||||||
|             line.push({ text: " ", formats: [] }); |  | ||||||
|          } |  | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|       newLine(); |  | ||||||
|  |  | ||||||
|       let msg: Message = { |       let msg: Message = { | ||||||
|          date: new Date(), |          date, | ||||||
|          file, |          file, | ||||||
|          name: this._name, |          names, | ||||||
|          text: { |          text: formattedMessage, | ||||||
|             raw, |  | ||||||
|             formatted, |  | ||||||
|          }, |  | ||||||
|          type, |          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 { |       for (const adapter of this.#adapters) { | ||||||
|    [colorSymbol]: Colors; |          const cnt = LoggingBase[InitialisedAdapters].get(adapter); | ||||||
|    value: any; |          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); | ||||||
|  |             } | ||||||
|  |          } | ||||||
|  |       } | ||||||
|    } |    } | ||||||
|  |  | ||||||
| export function withColor(color: Colors, value: any): ColorFormat { |  | ||||||
|    return { |  | ||||||
|       [colorSymbol]: color, |  | ||||||
|       value, |  | ||||||
|    }; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| function getStack() { | function getStack() { | ||||||
| @ -507,9 +459,11 @@ function getCallerFile() { | |||||||
|    return { file: undefined, line: 0 }; |    return { file: undefined, line: 0 }; | ||||||
| } | } | ||||||
|  |  | ||||||
| function getCallerFromExisting( | function getCallerFromExisting(err: Error): { | ||||||
|    err: Error |    file: string; | ||||||
| ): { file: string; line: number; column?: number } { |    line: number; | ||||||
|  |    column?: number; | ||||||
|  | } { | ||||||
|    if (!err || !err.stack) return { file: "NOFILE", line: 0 }; |    if (!err || !err.stack) return { file: "NOFILE", line: 0 }; | ||||||
|    let lines = err.stack.split("\n"); |    let lines = err.stack.split("\n"); | ||||||
|    lines.shift(); // removing first line |    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 { Colors } from "./index.js"; | ||||||
| import { | import { | ||||||
|    Adapter, |    Adapter, | ||||||
|    Message, |    Message, | ||||||
|    FormattedLine, |  | ||||||
|    TerminalFormats, |    TerminalFormats, | ||||||
|    FormatTypes, |    Formatted, | ||||||
|  |    IFormatted, | ||||||
|  |    LoggingTypes, | ||||||
| } from "./types.js"; | } from "./types.js"; | ||||||
|  |  | ||||||
| declare const Deno: any; | declare const Deno: any; | ||||||
|  | declare const process: any; | ||||||
|  |  | ||||||
| const browser = typeof window !== "undefined" && typeof Deno === "undefined"; | const browser = typeof window !== "undefined" && typeof Deno === "undefined"; | ||||||
|  | const deno = typeof Deno !== "undefined"; | ||||||
|  | const NodeJS = typeof process !== undefined; | ||||||
|  |  | ||||||
| export class ConsoleAdapter implements Adapter { | export class ConsoleAdapter implements Adapter { | ||||||
|  |    level: LoggingTypes = LoggingTypes.Debug; | ||||||
|    constructor(private colors: boolean = true) { } |    constructor(private colors: boolean = true) { } | ||||||
|  |  | ||||||
|    init(observable: ObservableInterface<Message>) { |    init() { } | ||||||
|       observable.subscribe(this.onMessage.bind(this)); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    flush() { } |    flush() { } | ||||||
|  |  | ||||||
|  |    setLevel(level: LoggingTypes) { | ||||||
|  |       this.level = level; | ||||||
|  |    } | ||||||
|    // TODO: Check if required! |    // TODO: Check if required! | ||||||
|    // private escape(text: string): string { |    // private escape(text: string): string { | ||||||
|    //    return text |    //    return text | ||||||
| @ -28,25 +32,29 @@ export class ConsoleAdapter implements Adapter { | |||||||
|    //       .replace(/%c/g, "%%c") |    //       .replace(/%c/g, "%%c") | ||||||
|    // } |    // } | ||||||
|  |  | ||||||
|    private formatLine(line: FormattedLine): [string, string[] | undefined] { |    private format( | ||||||
|  |       formatted: IFormatted<string>[] | ||||||
|  |    ): [string, string[] | undefined] { | ||||||
|       let text = ""; |       let text = ""; | ||||||
|       let style_formats: string[] = []; |       let style_formats: string[] = []; | ||||||
|  |  | ||||||
|       if (!browser) { |       if (!browser) { | ||||||
|          for (let part of line) { |          // NodeJS or Deno | ||||||
|  |          for (const format of formatted) { | ||||||
|             let formats = ""; |             let formats = ""; | ||||||
|             for (let format of part.formats) { |             if (format.bold) { | ||||||
|                switch (format.type) { |  | ||||||
|                   case FormatTypes.BOLD: |  | ||||||
|                formats += TerminalFormats.Bold; |                formats += TerminalFormats.Bold; | ||||||
|                      break; |             } | ||||||
|                   case FormatTypes.UNDERSCORE: |  | ||||||
|                      formats += TerminalFormats.Underscore; |             if (format.blink) { | ||||||
|                      break; |  | ||||||
|                   case FormatTypes.BLINK: |  | ||||||
|                formats += TerminalFormats.Blink; |                formats += TerminalFormats.Blink; | ||||||
|                      break; |             } | ||||||
|                   case FormatTypes.COLOR: |  | ||||||
|  |             if (format.underscore) { | ||||||
|  |                formats += TerminalFormats.Underscore; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (format.color) { | ||||||
|                switch (format.color) { |                switch (format.color) { | ||||||
|                   case Colors.RED: |                   case Colors.RED: | ||||||
|                      formats += TerminalFormats.FgRed; |                      formats += TerminalFormats.FgRed; | ||||||
| @ -70,49 +78,76 @@ export class ConsoleAdapter implements Adapter { | |||||||
|                      formats += TerminalFormats.FgWhite; |                      formats += TerminalFormats.FgWhite; | ||||||
|                      break; |                      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; |                      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 { |       } else { | ||||||
|          for (let part of line) { |          for (const format of formatted) { | ||||||
|             let styles: string[] = []; |             let styles: string[] = []; | ||||||
|             let resetStyles: string[] = []; |             let resetStyles: string[] = []; | ||||||
|             for (let format of part.formats) { |             if (format.bold) { | ||||||
|                switch (format.type) { |  | ||||||
|                   case FormatTypes.BOLD: |  | ||||||
|                styles.push("font-weight: bold;"); |                styles.push("font-weight: bold;"); | ||||||
|                resetStyles.push("font-weight: unset"); |                resetStyles.push("font-weight: unset"); | ||||||
|                      break; |             } | ||||||
|                   case FormatTypes.UNDERSCORE: |  | ||||||
|  |             if (format.underscore) { | ||||||
|                styles.push("text-decoration: underline"); |                styles.push("text-decoration: underline"); | ||||||
|                resetStyles.push("text-decoration: unset"); |                resetStyles.push("text-decoration: unset"); | ||||||
|                      break; |             } | ||||||
|                   case FormatTypes.BLINK: |  | ||||||
|  |             if (format.blink) { | ||||||
|                styles.push("text-decoration: blink"); |                styles.push("text-decoration: blink"); | ||||||
|                resetStyles.push("text-decoration: unset"); |                resetStyles.push("text-decoration: unset"); | ||||||
|                      break; |             } | ||||||
|                   case FormatTypes.COLOR: |  | ||||||
|  |             if (format.color) { | ||||||
|                let color = ""; |                let color = ""; | ||||||
|                switch (format.color) { |                switch (format.color) { | ||||||
|                   case Colors.RED: |                   case Colors.RED: | ||||||
|                            color = "red"; |                      color = "#ff5f5f"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.GREEN: |                   case Colors.GREEN: | ||||||
|                            color = "green"; |                      color = "#62ff5f"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.YELLOW: |                   case Colors.YELLOW: | ||||||
|                            color = "gold"; |                      color = "#ffea37"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.BLUE: |                   case Colors.BLUE: | ||||||
|                            color = "blue"; |                      color = "#379aff"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.MAGENTA: |                   case Colors.MAGENTA: | ||||||
|                            color = "magenta"; |                      color = "#f837ff"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.CYAN: |                   case Colors.CYAN: | ||||||
|                            color = "cyan"; |                      color = "#37e4ff"; | ||||||
|                      break; |                      break; | ||||||
|                   case Colors.WHITE: |                   case Colors.WHITE: | ||||||
|                      color = "white"; |                      color = "white"; | ||||||
| @ -120,10 +155,38 @@ export class ConsoleAdapter implements Adapter { | |||||||
|                } |                } | ||||||
|                styles.push("color: " + color); |                styles.push("color: " + color); | ||||||
|                resetStyles.push("color: unset"); |                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; |                      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(";")); |             style_formats.push(styles.join(";"), resetStyles.join(";")); | ||||||
|          } |          } | ||||||
|       } |       } | ||||||
| @ -132,34 +195,26 @@ export class ConsoleAdapter implements Adapter { | |||||||
|    } |    } | ||||||
|  |  | ||||||
|    onMessage(message: Message) { |    onMessage(message: Message) { | ||||||
|       let lines = message.text.formatted; |  | ||||||
|  |  | ||||||
|       let prefix = ""; |  | ||||||
|       if (message.name) prefix = `[${message.name}]=>`; |  | ||||||
|  |  | ||||||
|       if (browser) { |       if (browser) { | ||||||
|          if (this.colors) { |          if (this.colors) { | ||||||
|             let formats: string[] = []; |             const [text, formats] = this.format(message.text); | ||||||
|             let text = lines |  | ||||||
|                .map((line) => { |  | ||||||
|                   let [t, fmts] = this.formatLine(line); |  | ||||||
|                   formats.push(...fmts); |  | ||||||
|                   return prefix + t; |  | ||||||
|                }) |  | ||||||
|                .join("\n"); |  | ||||||
|             // console.log(formats); |  | ||||||
|             console.log(text, ...formats); |             console.log(text, ...formats); | ||||||
|          } else { |          } else { | ||||||
|             console.log(message.text.raw.join("\n")); |             console.log(Formatted.strip(message.text)); | ||||||
|          } |          } | ||||||
|       } else { |       } else { | ||||||
|          if (this.colors) { |          const text = this.colors | ||||||
|             lines.forEach((line) => { |             ? this.format(message.text)[0] | ||||||
|                let [text] = this.formatLine(line); |             : Formatted.strip(message.text); | ||||||
|                console.log(prefix + text); |  | ||||||
|             }); |          if (deno) { | ||||||
|  |             //TODO: Deno specific thing | ||||||
|  |             console.log(text); | ||||||
|  |          } else if (typeof process !== "undefined") { | ||||||
|  |             //NodeJS | ||||||
|  |             process.stdout.write(text + "\n"); | ||||||
|          } else { |          } 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"; | import { LoggingBase } from "./base.js"; | ||||||
|  | export { LoggingBase } from "./base.js"; | ||||||
| export { ConsoleAdapter } from "./consolewriter.js"; | export { ConsoleAdapter } from "./consolewriter.js"; | ||||||
| export { | export { ILoggingOptions, INativeFunctions } from "./base.js"; | ||||||
|    LoggingBase, |  | ||||||
|    LoggingBaseOptions, |  | ||||||
|    removeColors, |  | ||||||
|    withColor, |  | ||||||
| } from "./base.js"; |  | ||||||
| export { | export { | ||||||
|    Adapter, |    Adapter, | ||||||
|    LoggingTypes, |    LoggingTypes, | ||||||
|    Message, |    Message, | ||||||
|    FormatConfig, |    FormatConfig, | ||||||
|    FormattedLine, |  | ||||||
|    DefaultFormatConfig as DefaultColorMap, |    DefaultFormatConfig as DefaultColorMap, | ||||||
|    FormattedText, |    IColors as Colors, | ||||||
|    Colors, |  | ||||||
|    Format, |    Format, | ||||||
|    FormatTypes, |  | ||||||
|    TerminalFormats, |    TerminalFormats, | ||||||
|  |    Formatted, | ||||||
|  |    IFormatted, | ||||||
|  |    ILoggingInterface, | ||||||
|  |    ILoggingTimer, | ||||||
| } from "./types.js"; | } from "./types.js"; | ||||||
|  |  | ||||||
| export { ObservableInterface } from "@hibas123/utils"; | const Logging = new LoggingBase(); | ||||||
|  |  | ||||||
| export let Logging: LoggingBase = undefined; |  | ||||||
| Logging = new LoggingBase(); |  | ||||||
| export default Logging; | export default Logging; | ||||||
|  | |||||||
							
								
								
									
										49
									
								
								src/test.ts
									
									
									
									
									
								
							
							
						
						
									
										49
									
								
								src/test.ts
									
									
									
									
									
								
							| @ -1,26 +1,19 @@ | |||||||
| import { | import Logging from "./index.js"; | ||||||
|    Logging, |  | ||||||
|    LoggingBase, | import { LoggingBase, LoggingTypes, Format } from "./index.js"; | ||||||
|    LoggingTypes, |  | ||||||
|    Colors, |  | ||||||
|    withColor, |  | ||||||
| } from "./index.js"; |  | ||||||
|  |  | ||||||
| Logging.log("test"); | Logging.log("test"); | ||||||
| Logging.log("i", "am", { a: "an" }, 1000); | Logging.log("i", "am", { a: "an" }, 1000); | ||||||
| Logging.error(new Error("fehler 001")); | Logging.error(new Error("fehler 001")); | ||||||
| Logging.debug("Some Debug infos"); | Logging.debug("Some Debug infos"); | ||||||
| Logging.errorMessage("i", "am", "an", "error"); | Logging.error("i", "am", "an", "error"); | ||||||
|  |  | ||||||
| Logging.log( | Logging.log( | ||||||
|    "\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m TEST \x1b[31m\x1b[31m\x1b[31m" |    "\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m TEST \x1b[31m\x1b[31m\x1b[31m" | ||||||
| ); | ); | ||||||
|  |  | ||||||
| Logging.log( | Logging.log(Format.magenta("This text should be magenta!"), "This not!"); | ||||||
|    withColor(Colors.MAGENTA, "This text should be magenta!"), | Logging.log(Format.magenta({ somekey: "Some value" })); | ||||||
|    "This not!" |  | ||||||
| ); |  | ||||||
| Logging.log(withColor(Colors.MAGENTA, { somekey: "Some value" })); |  | ||||||
|  |  | ||||||
| let err = new Error(); | let err = new Error(); | ||||||
| if (typeof err.stack !== "string") console.log("Stacktrace invalid", err.stack); | 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("Hello from custom Logger"); | ||||||
| cus.log("This has some %c symbols inside of it!"); | 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"); | 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"); | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2"); | cus2.log("Hello from custom Logger 2"); | ||||||
| cus22.log("Hello from custom Logger 22"); | 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.warning("This should not be there 4"); | ||||||
|  |  | ||||||
| Logging.error("This should be there 1"); | 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"); | 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.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"); | c2.log("MSG from C2"); | ||||||
| c1.log("MSG from C1"); | c1.log("MSG from C1"); | ||||||
| Logging.log("MSG from root"); | Logging.log("MSG from root"); | ||||||
|  |  | ||||||
| const timer = Logging.time("timer1", "Test Timer"); | const timer = Logging.time("timer1"); | ||||||
| setTimeout(() => timer.end(), 1000); | setTimeout(() => timer.end(), 1000); | ||||||
|  |  | ||||||
|  | const withoutFile = new LoggingBase({ | ||||||
|  |    resolve_filename: false, | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | withoutFile.log("This should not have a file attached"); | ||||||
|  | |||||||
							
								
								
									
										236
									
								
								src/types.ts
									
									
									
									
									
								
							
							
						
						
									
										236
									
								
								src/types.ts
									
									
									
									
									
								
							| @ -1,9 +1,7 @@ | |||||||
| import { ObservableInterface } from "@hibas123/utils"; |  | ||||||
|  |  | ||||||
| export enum LoggingTypes { | export enum LoggingTypes { | ||||||
|    Debug, |    Debug, | ||||||
|    Log, |    Log, | ||||||
|    Warning, |    Warn, | ||||||
|    Error, |    Error, | ||||||
| } | } | ||||||
|  |  | ||||||
| @ -34,14 +32,7 @@ export const TerminalFormats = { | |||||||
|    BgWhite: "\x1b[47m", |    BgWhite: "\x1b[47m", | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export enum FormatTypes { | export enum IColors { | ||||||
|    COLOR, |  | ||||||
|    BOLD, |  | ||||||
|    UNDERSCORE, |  | ||||||
|    BLINK, |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export enum Colors { |  | ||||||
|    NONE, |    NONE, | ||||||
|    RED, |    RED, | ||||||
|    GREEN, |    GREEN, | ||||||
| @ -52,70 +43,194 @@ export enum Colors { | |||||||
|    WHITE, |    WHITE, | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export interface IFormatted<T = any> { | ||||||
|  |    color?: IColors; | ||||||
|  |    bgcolor?: IColors; | ||||||
|  |    bold?: boolean; | ||||||
|  |    italic?: boolean; | ||||||
|  |    blink?: boolean; | ||||||
|  |    underscore?: boolean; | ||||||
|  |  | ||||||
|  |    content: T; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | 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 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 { | export interface FormatConfig { | ||||||
|    error: Format[]; |    error: Formatted; | ||||||
|    warning: Format[]; |    warning: Formatted; | ||||||
|    log: Format[]; |    log: Formatted; | ||||||
|    debug: Format[]; |    debug: Formatted; | ||||||
|  |  | ||||||
|    date: Format[]; |    date: Formatted; | ||||||
|    file: Format[]; |    file: Formatted; | ||||||
|  |  | ||||||
|  |    names: Formatted; | ||||||
|  |    names_delimiter: Formatted; | ||||||
| } | } | ||||||
|  |  | ||||||
| function colorFormat(color: Colors) { |  | ||||||
|    return { |  | ||||||
|       type: FormatTypes.COLOR, |  | ||||||
|       color, |  | ||||||
|    }; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const boldFormat = { |  | ||||||
|    type: FormatTypes.BOLD, |  | ||||||
| }; |  | ||||||
|  |  | ||||||
| export class DefaultFormatConfig implements FormatConfig { | export class DefaultFormatConfig implements FormatConfig { | ||||||
|    error = [colorFormat(Colors.RED), boldFormat]; |    error = new Formatted()._color(IColors.RED)._bold(); | ||||||
|    warning = [colorFormat(Colors.YELLOW), boldFormat]; |    warning = new Formatted()._color(IColors.YELLOW)._bold(); | ||||||
|    log = [colorFormat(Colors.NONE), boldFormat]; |    log = new Formatted()._color(IColors.NONE)._bold(); | ||||||
|    debug = [colorFormat(Colors.CYAN), boldFormat]; |    debug = new Formatted()._color(IColors.CYAN)._bold(); | ||||||
|  |  | ||||||
|    date = [colorFormat(Colors.NONE)]; |    date = new Formatted()._color(IColors.NONE); | ||||||
|    file = [colorFormat(Colors.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 { | export interface Message { | ||||||
|    type: LoggingTypes; |    type: LoggingTypes; | ||||||
|    name?: string; |    names?: string[]; | ||||||
|    text: { |    text: IFormatted<string>[]; | ||||||
|       raw: string[]; |  | ||||||
|       formatted: FormattedLine[]; |  | ||||||
|    }; |  | ||||||
|    date: Date; |    date: Date; | ||||||
|    file: string; |    file: string; | ||||||
| } | } | ||||||
|  |  | ||||||
| export interface Adapter { | export interface Adapter { | ||||||
|  |    readonly level: LoggingTypes; | ||||||
|  |  | ||||||
|    /** |    /** | ||||||
|     * This function initialises the Adapter. It might be called multiple times, when added to multiple instances |     * This function initialises the Adapter. It might be called multiple times, when added to multiple instances | ||||||
|     * @param observable An observable to subscribe to messages |     * @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: true): void; | ||||||
|    flush(sync: false): void | Promise<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. |     * 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; |    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": { |    "compilerOptions": { | ||||||
|       "module": "commonjs", |       "module": "ESNext", | ||||||
|       "target": "esnext", |       "target": "ESNext", | ||||||
|  |       "moduleResolution": "node", | ||||||
|  |       "outDir": "esm", | ||||||
|       "noImplicitAny": false, |       "noImplicitAny": false, | ||||||
|       "sourceMap": true, |       "sourceMap": true, | ||||||
|       "outDir": "out", |       "declaration": true | ||||||
|       "declaration": true, |  | ||||||
|       "typeRoots": ["node_modules/@types"] |  | ||||||
|    }, |    }, | ||||||
|    "exclude": ["node_modules"], |    "exclude": ["node_modules"], | ||||||
|    "include": ["src"] |    "include": ["src"] | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user
	