Compare commits
	
		
			27 Commits
		
	
	
		
			9182efe7e7
			...
			4.0.2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 3526766e68 | |||
| 3de5f368ef | |||
| e0b51625d8 | |||
| 153aca0ccb | |||
| 7ca0c4fd72 | |||
| 8e183ac1a5 | |||
| feed4626e6 | |||
| 9d4e521619 | |||
| 7d75f65dd3 | |||
| eeed068ddd | |||
| 6daf815ea8 | |||
| 357b98c69a | |||
| 96d7808f35 | |||
| 176d37249d | |||
| bcff79fc90 | |||
| b92caf6468 | |||
| f34800d725 | |||
| 1fd8da459b | |||
| 4420fb13ea | |||
| 511bdf127f | |||
| 7ae1c3e16e | |||
| ca896c1c34 | |||
| 30f5e241ae | |||
| a1cd860688 | |||
| f51f4e2aad | |||
| fa7a168f17 | |||
| 2eb9356a9d | 
							
								
								
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -2,4 +2,4 @@ node_modules/ | ||||
| logs/ | ||||
| yarn.lock | ||||
| out/ | ||||
| .history/ | ||||
| esm/ | ||||
							
								
								
									
										17
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								.vscode/launch.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,17 @@ | ||||
| { | ||||
|    // Use IntelliSense to learn about possible attributes. | ||||
|    // Hover to view descriptions of existing attributes. | ||||
|    // For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 | ||||
|    "version": "0.2.0", | ||||
|    "configurations": [ | ||||
|       { | ||||
|          "type": "pwa-node", | ||||
|          "request": "launch", | ||||
|          "name": "Launch Program", | ||||
|          "skipFiles": ["<node_internals>/**"], | ||||
|          "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" | ||||
|    ] | ||||
| } | ||||
							
								
								
									
										4066
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										4066
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										30
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								package.json
									
									
									
									
									
								
							| @ -1,16 +1,17 @@ | ||||
| { | ||||
|    "name": "@hibas123/logging", | ||||
|    "version": "2.3.0", | ||||
|    "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": "tsc", | ||||
|       "prepublishOnly": "npm run build", | ||||
|       "build": "tsc", | ||||
|       "watch-ts": "tsc --watch", | ||||
|       "watch-js": "nodemon out/test.js", | ||||
|       "watch": "concurrently npm:watch-*", | ||||
|       "test": "node out/test.js" | ||||
|       "test": "tsc && node esm/test.js", | ||||
|       "bench": "tsc && node benchmark.js", | ||||
|       "prof": "tsc && ndb --prof benchmark.js" | ||||
|    }, | ||||
|    "repository": { | ||||
|       "type": "git", | ||||
| @ -20,16 +21,15 @@ | ||||
|    "license": "MIT", | ||||
|    "files": [ | ||||
|       "src/", | ||||
|       "out/", | ||||
|       "esm/", | ||||
|       "tsconfig.json", | ||||
|       "readme.md" | ||||
|    ], | ||||
|    "devDependencies": { | ||||
|       "concurrently": "^5.1.0", | ||||
|       "nodemon": "^2.0.2", | ||||
|       "typescript": "^3.8.3" | ||||
|    }, | ||||
|    "dependencies": { | ||||
|       "@hibas123/utils": "^2.2.3" | ||||
|       "concurrently": "^9.2.1", | ||||
|       "ndb": "^1.1.5", | ||||
|       "nodemon": "^3.1.10", | ||||
|       "ts-node": "^10.9.2", | ||||
|       "typescript": "^5.9.3" | ||||
|    } | ||||
| } | ||||
							
								
								
									
										687
									
								
								src/base.ts
									
									
									
									
									
								
							
							
						
						
									
										687
									
								
								src/base.ts
									
									
									
									
									
								
							| @ -1,37 +1,19 @@ | ||||
| import { Observable } from "@hibas123/utils"; | ||||
| import { ConsoleAdapter } from "./consolewriter"; | ||||
| import inspect from "./inspect"; | ||||
| import { ConsoleAdapter } from "./consolewriter.js"; | ||||
| import inspect from "./inspect.js"; | ||||
| import { | ||||
|    Adapter, | ||||
|    LoggingTypes, | ||||
|    Message, | ||||
|    FormatConfig, | ||||
|    DefaultFormatConfig, | ||||
|    Format, | ||||
|    FormatTypes, | ||||
|    Colors, | ||||
|    FormattedText, | ||||
|    FormattedLine, | ||||
| } from "./types"; | ||||
| import Logging from "."; | ||||
|    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 | ||||
|     */ | ||||
| @ -40,348 +22,405 @@ export interface LoggingBaseOptions { | ||||
|     * Prints output to console | ||||
|     */ | ||||
|    console: boolean; | ||||
|  | ||||
|    /** | ||||
|     * Enables printing of calling file | ||||
|     */ | ||||
|    resolve_filename: boolean; | ||||
| } | ||||
|  | ||||
| const adapterCache = new WeakMap<Adapter, number>(); | ||||
| export class LoggingBase { | ||||
|    private _formatMap: FormatConfig = new DefaultFormatConfig(); | ||||
| export interface INativeFunctions { | ||||
|    startTimer(): any; | ||||
|    diffTime(start: any, end: any): number; | ||||
|    endTimer(start: any): number; | ||||
| } | ||||
|  | ||||
|    public set formatMap(value: FormatConfig) { | ||||
|       this._formatMap = value; | ||||
|    } | ||||
|  | ||||
|    private adapter = new Set<Adapter>(); | ||||
|    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) { | ||||
|       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 (config.console) { | ||||
|          this.addAdapter(new 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); | ||||
|    } | ||||
|  | ||||
|    addAdapter(adapter: Adapter) { | ||||
|       if (!this.adapter.has(adapter)) { | ||||
|          this.adapter.add(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); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|    flush(sync: true): void; | ||||
|    flush(sync: false): Promise<void>; | ||||
|    flush(sync: boolean): void | Promise<void> { | ||||
|       if (sync) { | ||||
|          this.adapter.forEach((elm) => elm.flush(true)); | ||||
|       } else { | ||||
|          let adapters: (void | Promise<void>)[] = []; | ||||
|          this.adapter.forEach((elm) => adapters.push(elm.flush(false))); | ||||
|          return Promise.all(adapters).then(() => {}); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|    private $closed = false; | ||||
|  | ||||
|    public close() { | ||||
|       if (this.$closed) return; | ||||
|       this.$closed = true; | ||||
|  | ||||
|       this.adapter.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.adapter = 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); | ||||
|          Logging.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 } | ||||
|       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_raw = caller || getCallerFile(); | ||||
|       let file = `${file_raw.file}:${String(file_raw.line).padEnd(3, " ")}`; | ||||
|       let file: string | undefined = undefined; | ||||
|       if (this.#resolve_filename) { | ||||
|          let file_raw = caller; | ||||
|          if (!file_raw) { | ||||
|             try { | ||||
|                file_raw = getCallerFile(); | ||||
|             } catch (err) { | ||||
|                file_raw = { | ||||
|                   file: "<unknown>", | ||||
|                   line: 0, | ||||
|                   column: 0, | ||||
|                }; | ||||
|             } | ||||
|          } | ||||
|  | ||||
|          if ( | ||||
|             PROJECT_ROOT && | ||||
|             file_raw.file && | ||||
|             file_raw.file.startsWith(PROJECT_ROOT) | ||||
|          ) { | ||||
|             let newF = file_raw.file.substring(PROJECT_ROOT.length); | ||||
|  | ||||
|             if (newF.startsWith("/") || newF.startsWith("\\")) | ||||
|                newF = newF.substring(1); | ||||
|  | ||||
|             file_raw.file = newF; | ||||
|          } | ||||
|          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() { | ||||
|    // Save original Error.prepareStackTrace | ||||
|    let origPrepareStackTrace = (<any>Error).prepareStackTrace; | ||||
|  | ||||
|    try { | ||||
|       // Override with function that just returns `stack` | ||||
|       (<any>Error).prepareStackTrace = function (_, stack) { | ||||
|          return stack; | ||||
| @ -393,17 +432,11 @@ function getStack() { | ||||
|       // Evaluate `err.stack`, which calls our new `Error.prepareStackTrace` | ||||
|       let stack: any[] = <any>err.stack; | ||||
|  | ||||
|       return stack; | ||||
|    } finally { | ||||
|       // Restore original `Error.prepareStackTrace` | ||||
|       (<any>Error).prepareStackTrace = origPrepareStackTrace; | ||||
|  | ||||
|    // Remove superfluous function call on stack | ||||
|    stack.shift(); // getStack --> Error | ||||
|  | ||||
|    return stack; | ||||
| } | ||||
|  | ||||
| function baseName(path) { | ||||
|    return path.split(/[\\/]/).pop(); | ||||
|    } | ||||
| } | ||||
|  | ||||
| function getCallerFile() { | ||||
| @ -414,29 +447,57 @@ function getCallerFile() { | ||||
|  | ||||
|       while (stack.length) { | ||||
|          let caller_file = stack.shift(); | ||||
|          if (current_file !== caller_file.getFileName()) | ||||
|          if (current_file !== caller_file.getFileName()) { | ||||
|             return { | ||||
|                file: baseName(caller_file.getFileName()), | ||||
|                file: caller_file.getFileName(), | ||||
|                line: caller_file.getLineNumber(), | ||||
|                column: caller_file.getColumnNumber(), | ||||
|             }; | ||||
|          } | ||||
|    } catch (err) {} | ||||
|       } | ||||
|    } catch (err) { } | ||||
|    return { file: undefined, line: 0 }; | ||||
| } | ||||
|  | ||||
| function getCallerFromExisting(err: Error): { file: string; line: 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 | ||||
|    while (lines.length > 0) { | ||||
|       let line = lines.shift(); | ||||
|       let matches = line.match(/[a-zA-Z_-]+[.][a-zA-Z_-]+[:][0-9]+/g); | ||||
|       let matches = line.match( | ||||
|          /[<]?([a-zA-Z]:)?([\/\\]?[a-zA-Z_-])+[.>][a-zA-Z_-]*([:][0-9]+)+/g | ||||
|       ); | ||||
|       if (matches && matches.length > 0) { | ||||
|          let [f, line] = matches[0].split(":"); | ||||
|          let match = matches[0].trim(); | ||||
|          let locationString = match.match(/([:][0-9]+)+$/gm)[0]; | ||||
|          let line: number; | ||||
|          let column: number; | ||||
|          if (locationString) { | ||||
|             match = match.slice(0, match.length - locationString.length); | ||||
|             locationString = locationString.substring(1); | ||||
|             [line, column] = locationString.split(":").map(Number); | ||||
|          } | ||||
|          let file = match; | ||||
|          return { | ||||
|             file: f, | ||||
|             line: Number(line), | ||||
|             file, | ||||
|             line, | ||||
|             column, | ||||
|          }; | ||||
|       } | ||||
|    } | ||||
| } | ||||
|  | ||||
| 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,22 +1,30 @@ | ||||
| import { ObservableInterface } from "@hibas123/utils"; | ||||
| import { Colors } from "./index"; | ||||
| import { Colors } from "./index.js"; | ||||
| import { | ||||
|    Adapter, | ||||
|    Message, | ||||
|    FormattedLine, | ||||
|    TerminalFormats, | ||||
|    FormatTypes, | ||||
| } from "./types"; | ||||
|    Formatted, | ||||
|    IFormatted, | ||||
|    LoggingTypes, | ||||
| } from "./types.js"; | ||||
|  | ||||
| const browser = typeof window !== "undefined"; | ||||
| 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 { | ||||
|    init(observable: ObservableInterface<Message>) { | ||||
|       observable.subscribe(this.onMessage.bind(this)); | ||||
|    level: LoggingTypes = LoggingTypes.Debug; | ||||
|    constructor(private colors: boolean = true) { } | ||||
|  | ||||
|    init() { } | ||||
|    flush() { } | ||||
|  | ||||
|    setLevel(level: LoggingTypes) { | ||||
|       this.level = level; | ||||
|    } | ||||
|  | ||||
|    flush() {} | ||||
|  | ||||
|    // TODO: Check if required! | ||||
|    // private escape(text: string): string { | ||||
|    //    return text | ||||
| @ -24,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; | ||||
| @ -66,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"; | ||||
| @ -116,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(";")); | ||||
|          } | ||||
|       } | ||||
| @ -128,27 +195,27 @@ export class ConsoleAdapter implements Adapter { | ||||
|    } | ||||
|  | ||||
|    onMessage(message: Message) { | ||||
|       let lines = message.text.formatted; | ||||
|  | ||||
|       let prefix = ""; | ||||
|       if (message.name) prefix = `[${message.name}]=>`; | ||||
|  | ||||
|       if (browser) { | ||||
|          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); | ||||
|          if (this.colors) { | ||||
|             const [text, formats] = this.format(message.text); | ||||
|             console.log(text, ...formats); | ||||
|          } else { | ||||
|          lines.forEach((line) => { | ||||
|             let [text] = this.formatLine(line); | ||||
|             console.log(prefix + text); | ||||
|          }); | ||||
|             console.log(Formatted.strip(message.text)); | ||||
|          } | ||||
|       } else { | ||||
|          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 { | ||||
|             console.log(text); | ||||
|          } | ||||
|       } | ||||
|    } | ||||
| } | ||||
|  | ||||
							
								
								
									
										28
									
								
								src/index.ts
									
									
									
									
									
								
							
							
						
						
									
										28
									
								
								src/index.ts
									
									
									
									
									
								
							| @ -1,27 +1,23 @@ | ||||
| import { LoggingBase } from "./base"; | ||||
| export { ConsoleAdapter } from "./consolewriter"; | ||||
| export { | ||||
|    LoggingBase, | ||||
|    LoggingBaseOptions, | ||||
|    removeColors, | ||||
|    withColor, | ||||
| } from "./base"; | ||||
| import { LoggingBase } from "./base.js"; | ||||
| export { LoggingBase } from "./base.js"; | ||||
| export { ConsoleAdapter } from "./consolewriter.js"; | ||||
| export { ILoggingOptions, INativeFunctions } from "./base.js"; | ||||
|  | ||||
| export { | ||||
|    Adapter, | ||||
|    LoggingTypes, | ||||
|    Message, | ||||
|    FormatConfig, | ||||
|    FormattedLine, | ||||
|    DefaultFormatConfig as DefaultColorMap, | ||||
|    FormattedText, | ||||
|    Colors, | ||||
|    IColors as Colors, | ||||
|    Format, | ||||
|    FormatTypes, | ||||
|    TerminalFormats, | ||||
| } from "./types"; | ||||
|    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; | ||||
|  | ||||
							
								
								
									
										41
									
								
								src/test.ts
									
									
									
									
									
								
							
							
						
						
									
										41
									
								
								src/test.ts
									
									
									
									
									
								
							| @ -1,20 +1,19 @@ | ||||
| import { Logging, LoggingBase, LoggingTypes, Colors, withColor } from "."; | ||||
| 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); | ||||
| @ -23,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"); | ||||
| @ -47,7 +46,27 @@ 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"); | ||||
|  | ||||
| const timer = Logging.time("timer1", "Test Timer"); | ||||
| Logging.logLevel = LoggingTypes.Debug; | ||||
|  | ||||
| const c1 = Logging.getChild("child-level-1"); | ||||
|  | ||||
| c1.log("Hello from Child 1"); | ||||
|  | ||||
| const c2 = c1.getChild("child-level-2"); | ||||
|  | ||||
| c2.log("Hello from Child 2"); | ||||
|  | ||||
| c2.log("MSG from C2"); | ||||
| c1.log("MSG from C1"); | ||||
| Logging.log("MSG from root"); | ||||
|  | ||||
| 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,19 +1,13 @@ | ||||
| { | ||||
|    "compilerOptions": { | ||||
|       "module": "commonjs", | ||||
|       "target": "es2017", | ||||
|       "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" | ||||
|    ] | ||||
|    "exclude": ["node_modules"], | ||||
|    "include": ["src"] | ||||
| } | ||||
		Reference in New Issue
	
	Block a user
	