Compare commits
	
		
			39 Commits
		
	
	
		
			51785e2b13
			...
			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 | |||
| 9182efe7e7 | |||
| 94d9731cdd | |||
| b647b8fae6 | |||
| facb7e7b40 | |||
| ed35aa17f2 | |||
| cedfe2cced | |||
| ce9742a20e | |||
| 675ec50139 | |||
| d2da8da690 | |||
| 0573b45429 | |||
| 475b787612 | |||
| cc5fc5ae08 | 
| @ -1,3 +1,4 @@ | ||||
| [*] | ||||
| charset = utf-8 | ||||
| indent_style = space | ||||
| indent_size = 3 | ||||
|  | ||||
							
								
								
									
										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; | ||||
| 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") | ||||
| 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. | ||||
| 
 | ||||
| @ -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. | ||||
| 
 | ||||
| ```javascript | ||||
|    const CustomLogging = new LoggingBase(name | { | ||||
| 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. | ||||
| 
 | ||||
| @ -52,7 +54,9 @@ The adapters need to provide a very simple Interface: | ||||
| 
 | ||||
| ```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" | ||||
|    ] | ||||
| } | ||||
							
								
								
									
										5860
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										5860
									
								
								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.0.8", | ||||
|    "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.0.0", | ||||
|       "nodemon": "^1.19.4", | ||||
|       "typescript": "^3.7.2" | ||||
|    }, | ||||
|    "dependencies": { | ||||
|       "@hibas123/utils": "^2.1.1" | ||||
|       "concurrently": "^9.2.1", | ||||
|       "ndb": "^1.1.5", | ||||
|       "nodemon": "^3.1.10", | ||||
|       "ts-node": "^10.9.2", | ||||
|       "typescript": "^5.9.3" | ||||
|    } | ||||
| } | ||||
							
								
								
									
										574
									
								
								src/base.ts
									
									
									
									
									
								
							
							
						
						
									
										574
									
								
								src/base.ts
									
									
									
									
									
								
							| @ -1,253 +1,430 @@ | ||||
| import { Observable } from "@hibas123/utils"; | ||||
| import { ConsoleAdapter } from "./consolewriter"; | ||||
| import inspect from "./inspect"; | ||||
| import { Adapter, LoggingTypes, Message, FormatConfig, DefaultFormatConfig, Format, FormatTypes, Colors, FormattedText, FormattedLine } from "./types"; | ||||
| import { ConsoleAdapter } from "./consolewriter.js"; | ||||
| import inspect from "./inspect.js"; | ||||
| import { | ||||
|    Adapter, | ||||
|    DefaultFormatConfig, | ||||
|    Format, | ||||
|    FormatConfig, | ||||
|    Formatted, | ||||
|    ILoggingInterface, | ||||
|    LoggingTypes, | ||||
|    Message, | ||||
| } from "./types.js"; | ||||
|  | ||||
| export function removeColors(text: string) { | ||||
|     text = text.replace(/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g, ""); | ||||
| const browser = typeof window !== "undefined"; | ||||
|  | ||||
|     // 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 | ||||
|     */ | ||||
|     name: string, | ||||
|    name: string; | ||||
|    /** | ||||
|     * Prints output to console | ||||
|     */ | ||||
|    console: boolean; | ||||
|  | ||||
|    /** | ||||
|     * Enables printing of calling file | ||||
|     */ | ||||
|    resolve_filename: boolean; | ||||
| } | ||||
|  | ||||
| export class LoggingBase { | ||||
|  | ||||
|     private _formatMap: FormatConfig = new DefaultFormatConfig(); | ||||
|  | ||||
|     public set formatMap(value: FormatConfig) { | ||||
|         this._formatMap = value; | ||||
| export interface INativeFunctions { | ||||
|    startTimer(): any; | ||||
|    diffTime(start: any, end: any): number; | ||||
|    endTimer(start: any): number; | ||||
| } | ||||
|  | ||||
|  | ||||
|     private adapter: Adapter[] = []; | ||||
|     private adapter_init: Promise<void>[] = []; | ||||
|  | ||||
|     private messageObservable = new Observable<Message>(); | ||||
|     protected _name: string; | ||||
|  | ||||
|     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 }; | ||||
| export const DefaultNativeFunctions = { | ||||
|    startTimer: () => { | ||||
|       if (browser && window.performance && window.performance.now) { | ||||
|          return window.performance.now(); | ||||
|       } else { | ||||
|             opt = options; | ||||
|          return Date.now(); | ||||
|       } | ||||
|  | ||||
|         let config = { | ||||
|             name: undefined, | ||||
|             console: true, | ||||
|             files: 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()); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     addAdapter(adapter: Adapter) { | ||||
|         this.adapter.push(adapter); | ||||
|         let prms = Promise.resolve(adapter.init(this.messageObservable.getPublicApi(), this._name)); | ||||
|         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)); | ||||
|    }, | ||||
|    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 Promise.all(this.adapter.map(elm => elm.flush(false))).then(() => { }); | ||||
|          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]; | ||||
|    } | ||||
|  | ||||
|     public close() { | ||||
|         this.adapter.forEach(adapter => adapter.close ? adapter.close() : undefined); | ||||
|     } | ||||
|    protected abstract message( | ||||
|       type: LoggingTypes, | ||||
|       names: string[], | ||||
|       message: any[], | ||||
|       caller?: { file: string; line: number; column?: number } | ||||
|    ): void; | ||||
|  | ||||
|     public waitForSetup() { | ||||
|         return Promise.all(this.adapter_init); | ||||
|    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); | ||||
|          } | ||||
|       } | ||||
|    } | ||||
|  | ||||
|    debug(...message: any[]) { | ||||
|         this.message(LoggingTypes.Debug, message); | ||||
|       this.message(LoggingTypes.Debug, this.#names, message); | ||||
|    } | ||||
|  | ||||
|    log(...message: any[]) { | ||||
|         this.message(LoggingTypes.Log, message); | ||||
|       this.message(LoggingTypes.Log, this.#names, message); | ||||
|    } | ||||
|  | ||||
|    warning(...message: any[]) { | ||||
|         this.message(LoggingTypes.Warning, message); | ||||
|       this.message(LoggingTypes.Warn, this.#names, message); | ||||
|    } | ||||
|  | ||||
|    warn(...message: any[]) { | ||||
|         this.message(LoggingTypes.Warning, message); | ||||
|       this.message(LoggingTypes.Warn, this.#names, message); | ||||
|    } | ||||
|  | ||||
|     error(error: Error | string) { | ||||
|         if (!error) error = "Empty ERROR was passed, so no informations available"; | ||||
|    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.message(LoggingTypes.Error, [error, "\n", e.stack]); | ||||
|          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, [error.message, "\n", error.stack], getCallerFromExisting(error)); | ||||
|          this.message( | ||||
|             LoggingTypes.Error, | ||||
|             this.#names, | ||||
|             [error.message, "\n", error.stack, "\n", ...message], | ||||
|             getCallerFromExisting(error) | ||||
|          ); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|     errorMessage(...message: any[]) { | ||||
|         this.message(LoggingTypes.Error, message); | ||||
|    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), | ||||
|       }; | ||||
|    } | ||||
|  | ||||
|     private message(type: LoggingTypes, message: any[], caller?: { file: string, line: number }) { | ||||
|         let date = new Date().toISOString().replace(/T/, ' ').replace(/\..+/, ''); | ||||
|    timeEnd(id: string) { | ||||
|       let timer = this.#timerMap.get(id); | ||||
|       if (timer) { | ||||
|          let end_time = LoggingBase.nativeFunctions.startTimer(); | ||||
|          let diff = LoggingBase.nativeFunctions.diffTime(timer.start, end_time); | ||||
|  | ||||
|         let file_raw = caller || getCallerFile(); | ||||
|         let file = `${file_raw.file}:${String(file_raw.line).padEnd(3, " ")}`; | ||||
|          this.message(LoggingTypes.Debug, this.#names, [ | ||||
|             Format.green(`[${timer.name}]`), | ||||
|             `->`, | ||||
|             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; | ||||
|    } | ||||
|  | ||||
|    abstract getChild(name: string): ILoggingInterface; | ||||
| } | ||||
|  | ||||
| export class LoggingBase extends LoggingInterface { | ||||
|    private static [InitialisedAdapters] = new Map<Adapter, number>(); | ||||
|    public static nativeFunctions: INativeFunctions = DefaultNativeFunctions; | ||||
|  | ||||
|    static DecoupledLogging = class extends LoggingInterface { | ||||
|       #lg: LoggingBase; | ||||
|       constructor(names: string[], lg: LoggingBase) { | ||||
|          super(names); | ||||
|          this.#lg = lg; | ||||
|       } | ||||
|  | ||||
|       message(...params: [any, any, any, any]) { | ||||
|          this.#lg.message(...params); | ||||
|       } | ||||
|  | ||||
|       getChild(name: string) { | ||||
|          return new LoggingBase.DecoupledLogging( | ||||
|             [...this.names, name], | ||||
|             this.#lg | ||||
|          ); | ||||
|       } | ||||
|    }; | ||||
|  | ||||
|    #closed = false; | ||||
|    #logLevel = LoggingTypes.Debug; | ||||
|    #resolve_filename: boolean; | ||||
|    #format_map: FormatConfig = new DefaultFormatConfig(); | ||||
|    #adapters = new Set<Adapter>(); | ||||
|  | ||||
|    set logLevel(level: LoggingTypes) { | ||||
|       this.#logLevel = level; | ||||
|    } | ||||
|  | ||||
|    get logLevel() { | ||||
|       return this.#logLevel; | ||||
|    } | ||||
|  | ||||
|    constructor(options?: Partial<ILoggingOptions>) { | ||||
|       super(options?.name ? [options.name] : []); | ||||
|       options = { | ||||
|          console: true, | ||||
|          resolve_filename: true, | ||||
|          ...options, | ||||
|       }; | ||||
|  | ||||
|       if (options.console) { | ||||
|          this.addAdapter(consoleAdapter); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|    async addAdapter(adapter: Adapter) { | ||||
|       const init = adapter.init(); | ||||
|  | ||||
|       const add = () => { | ||||
|          this.#adapters.add(adapter); | ||||
|          if (LoggingBase[InitialisedAdapters].has(adapter)) { | ||||
|             LoggingBase[InitialisedAdapters].set( | ||||
|                adapter, | ||||
|                LoggingBase[InitialisedAdapters].get(adapter) + 1 | ||||
|             ); | ||||
|          } else { | ||||
|             LoggingBase[InitialisedAdapters].set(adapter, 1); | ||||
|          } | ||||
|       }; | ||||
|  | ||||
|       if (!init) { | ||||
|          add(); | ||||
|       } else { | ||||
|          await Promise.resolve(init).then(add); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|    getChild(name: string): ILoggingInterface { | ||||
|       return new LoggingBase.DecoupledLogging([...this.names, name], this); | ||||
|    } | ||||
|  | ||||
|    protected message( | ||||
|       type: LoggingTypes, | ||||
|       names: string[], | ||||
|       message: any[], | ||||
|       caller?: { file: string; line: number; column?: number } | ||||
|    ) { | ||||
|       if (this.#logLevel > type) return; | ||||
|       if (this.#closed) { | ||||
|          //TODO: Maybe error? | ||||
|          return; | ||||
|       } | ||||
|  | ||||
|       const date = new Date(); | ||||
|       const isoStr = date.toISOString(); | ||||
|       const date_str = isoStr.substring(0, 10) + " " + isoStr.substring(11, 19); | ||||
|  | ||||
|       let file: string | undefined = undefined; | ||||
|       if (this.#resolve_filename) { | ||||
|          let file_raw = caller; | ||||
|          if (!file_raw) { | ||||
|             try { | ||||
|                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 | ||||
|             }) | ||||
|       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); | ||||
|             } | ||||
|          } | ||||
|          nameFormatted.push(new Formatted("]")); | ||||
|       } | ||||
|  | ||||
|         a("["); | ||||
|         a(date, this._formatMap.date); | ||||
|         a("]["); | ||||
|         a(type_str, type_format); | ||||
|         if (file_raw.file) { | ||||
|             a("]["); | ||||
|             a(file, this._formatMap.file); | ||||
|         } | ||||
|         a("]: "); | ||||
|       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 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 (typeof e === "object") { | ||||
|                     if (e[colorSymbol]) { | ||||
|                         formats.push({ | ||||
|                             type: FormatTypes.COLOR, | ||||
|                             color: e[colorSymbol] | ||||
|                         }) | ||||
|                         e = e.value; | ||||
|                     } | ||||
|                 } | ||||
|                 if (typeof e !== "string") | ||||
|                     e = inspect(e, { colors: true, showHidden: true, depth: 3 }) as string; | ||||
|       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(); | ||||
|          } | ||||
|  | ||||
|             removeColors(e).split("\n").map((text, index, { length }) => { | ||||
|                 line.push({ text, formats }); | ||||
|                 if (index < length - 1) { | ||||
|                     newLine(); | ||||
|          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; | ||||
|          } | ||||
|             }) | ||||
|  | ||||
|             if (!e.endsWith("\n") && i < message.length - 1) { | ||||
|                 line.push({ text: " ", formats: [] }); | ||||
|          // 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)); | ||||
|             } | ||||
|          }); | ||||
|  | ||||
|         newLine(); | ||||
|          formattedMessage.push(new Formatted(" ")); | ||||
|       }); | ||||
|  | ||||
|       let msg: Message = { | ||||
|             date: new Date(), | ||||
|          date, | ||||
|          file, | ||||
|             name: this._name, | ||||
|             text: { | ||||
|                 raw, | ||||
|                 formatted | ||||
|             }, | ||||
|             type | ||||
|          names, | ||||
|          text: formattedMessage, | ||||
|          type, | ||||
|       }; | ||||
|  | ||||
|       this.#adapters.forEach((adapter) => { | ||||
|          if (adapter.level <= type) { | ||||
|             adapter.onMessage(msg); | ||||
|          } | ||||
|       }); | ||||
|    } | ||||
|  | ||||
|         this.messageObservable.send(msg); | ||||
|    async close() { | ||||
|       if (this.#closed) return; | ||||
|       this.#closed = true; | ||||
|  | ||||
|       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); | ||||
|             } | ||||
|          } | ||||
|  | ||||
| const colorSymbol = Symbol("color"); | ||||
|  | ||||
| export interface ColorFormat { | ||||
|     [colorSymbol]: Colors; | ||||
|     value: any; | ||||
|       } | ||||
|  | ||||
| export function withColor(color: Colors, value: any): ColorFormat { | ||||
|     return { | ||||
|         [colorSymbol]: color, | ||||
|         value | ||||
|    } | ||||
| } | ||||
|  | ||||
| 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 | ||||
|     } | ||||
|          return stack; | ||||
|       }; | ||||
|  | ||||
|       // Create a new `Error`, which automatically gets `stack` | ||||
|       let err = new Error(); | ||||
| @ -255,49 +432,72 @@ 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() { | ||||
|    try { | ||||
|         let stack = getStack() | ||||
|       let stack = getStack(); | ||||
|  | ||||
|       let current_file = stack.shift().getFileName(); | ||||
|  | ||||
|       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()), | ||||
|                     line: caller_file.getLineNumber() | ||||
|                file: caller_file.getFileName(), | ||||
|                line: caller_file.getLineNumber(), | ||||
|                column: caller_file.getColumnNumber(), | ||||
|             }; | ||||
|          } | ||||
|       } | ||||
|    } 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,16 +1,30 @@ | ||||
| import { ObservableInterface } from "@hibas123/utils"; | ||||
| import { Colors } from "./index"; | ||||
| import { Adapter, Message, FormattedLine, TerminalFormats, FormatTypes } from "./types"; | ||||
| import { Colors } from "./index.js"; | ||||
| import { | ||||
|    Adapter, | ||||
|    Message, | ||||
|    TerminalFormats, | ||||
|    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; | ||||
|    } | ||||
|    // TODO: Check if required! | ||||
|    // private escape(text: string): string { | ||||
|    //    return text | ||||
| @ -18,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; | ||||
| @ -60,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"; | ||||
| @ -110,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(";")); | ||||
|          } | ||||
|       } | ||||
| @ -122,26 +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 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); | ||||
|          } | ||||
|       } | ||||
|    } | ||||
| } | ||||
							
								
								
									
										25
									
								
								src/index.ts
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								src/index.ts
									
									
									
									
									
								
							| @ -1,22 +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"; | ||||
|    TerminalFormats, | ||||
|    Formatted, | ||||
|    IFormatted, | ||||
|    ILoggingInterface, | ||||
|    ILoggingTimer, | ||||
| } from "./types.js"; | ||||
|  | ||||
| export { ObservableInterface } from "@hibas123/utils"; | ||||
| const Logging = new LoggingBase(); | ||||
|  | ||||
| export let Logging: LoggingBase = undefined; | ||||
| Logging = new LoggingBase(); | ||||
| export default Logging; | ||||
|  | ||||
							
								
								
									
										232
									
								
								src/inspect.ts
									
									
									
									
									
								
							
							
						
						
									
										232
									
								
								src/inspect.ts
									
									
									
									
									
								
							| @ -1,4 +1,3 @@ | ||||
|  | ||||
| /** | ||||
|  * Module exports. | ||||
|  */ | ||||
| @ -26,7 +25,7 @@ export default function inspect(obj: any, opts: Partial<InspectOptions>) { | ||||
|       depth: undefined, | ||||
|       colors: undefined, | ||||
|       showHidden: undefined, | ||||
|         customInspect: undefined | ||||
|       customInspect: undefined, | ||||
|    }; | ||||
|    // legacy... | ||||
|    if (arguments.length >= 3) ctx.depth = arguments[2]; | ||||
| @ -49,32 +48,32 @@ export default function inspect(obj: any, opts: Partial<InspectOptions>) { | ||||
|  | ||||
| // http://en.wikipedia.org/wiki/ANSI_escape_code#graphics | ||||
| inspect.colors = { | ||||
|     'bold': [1, 22], | ||||
|     'italic': [3, 23], | ||||
|     'underline': [4, 24], | ||||
|     'inverse': [7, 27], | ||||
|     'white': [37, 39], | ||||
|     'grey': [90, 39], | ||||
|     'black': [30, 39], | ||||
|     'blue': [34, 39], | ||||
|     'cyan': [36, 39], | ||||
|     'green': [32, 39], | ||||
|     'magenta': [35, 39], | ||||
|     'red': [31, 39], | ||||
|     'yellow': [33, 39] | ||||
|    bold: [1, 22], | ||||
|    italic: [3, 23], | ||||
|    underline: [4, 24], | ||||
|    inverse: [7, 27], | ||||
|    white: [37, 39], | ||||
|    grey: [90, 39], | ||||
|    black: [30, 39], | ||||
|    blue: [34, 39], | ||||
|    cyan: [36, 39], | ||||
|    green: [32, 39], | ||||
|    magenta: [35, 39], | ||||
|    red: [31, 39], | ||||
|    yellow: [33, 39], | ||||
| }; | ||||
|  | ||||
| // Don't use 'blue' not visible on cmd.exe | ||||
| inspect.styles = { | ||||
|     'special': 'cyan', | ||||
|     'number': 'yellow', | ||||
|     'boolean': 'yellow', | ||||
|     'undefined': 'grey', | ||||
|     'null': 'bold', | ||||
|     'string': 'green', | ||||
|     'date': 'magenta', | ||||
|    special: "cyan", | ||||
|    number: "yellow", | ||||
|    boolean: "yellow", | ||||
|    undefined: "grey", | ||||
|    null: "bold", | ||||
|    string: "green", | ||||
|    date: "magenta", | ||||
|    // "name": intentionally not styling | ||||
|     'regexp': 'red' | ||||
|    regexp: "red", | ||||
| }; | ||||
|  | ||||
| function stylizeNoColor(str, styleType) { | ||||
| @ -82,7 +81,7 @@ function stylizeNoColor(str, styleType) { | ||||
| } | ||||
|  | ||||
| function isBoolean(arg) { | ||||
|     return typeof arg === 'boolean'; | ||||
|    return typeof arg === "boolean"; | ||||
| } | ||||
|  | ||||
| function isUndefined(arg) { | ||||
| @ -93,23 +92,30 @@ function stylizeWithColor(str, styleType) { | ||||
|    var style = inspect.styles[styleType]; | ||||
|  | ||||
|    if (style) { | ||||
|         return '\u001b[' + inspect.colors[style][0] + 'm' + str + | ||||
|             '\u001b[' + inspect.colors[style][1] + 'm'; | ||||
|       return ( | ||||
|          "\u001b[" + | ||||
|          inspect.colors[style][0] + | ||||
|          "m" + | ||||
|          str + | ||||
|          "\u001b[" + | ||||
|          inspect.colors[style][1] + | ||||
|          "m" | ||||
|       ); | ||||
|    } else { | ||||
|       return str; | ||||
|    } | ||||
| } | ||||
|  | ||||
| function isFunction(arg) { | ||||
|     return typeof arg === 'function'; | ||||
|    return typeof arg === "function"; | ||||
| } | ||||
|  | ||||
| function isString(arg) { | ||||
|     return typeof arg === 'string'; | ||||
|    return typeof arg === "string"; | ||||
| } | ||||
|  | ||||
| function isNumber(arg) { | ||||
|     return typeof arg === 'number'; | ||||
|    return typeof arg === "number"; | ||||
| } | ||||
|  | ||||
| function isNull(arg) { | ||||
| @ -121,20 +127,22 @@ function hasOwn(obj, prop) { | ||||
| } | ||||
|  | ||||
| function isRegExp(re) { | ||||
|     return isObject(re) && objectToString(re) === '[object RegExp]'; | ||||
|    return isObject(re) && objectToString(re) === "[object RegExp]"; | ||||
| } | ||||
|  | ||||
| function isObject(arg) { | ||||
|     return typeof arg === 'object' && arg !== null; | ||||
|    return typeof arg === "object" && arg !== null; | ||||
| } | ||||
|  | ||||
| function isError(e) { | ||||
|     return isObject(e) && | ||||
|         (objectToString(e) === '[object Error]' || e instanceof Error); | ||||
|    return ( | ||||
|       isObject(e) && | ||||
|       (objectToString(e) === "[object Error]" || e instanceof Error) | ||||
|    ); | ||||
| } | ||||
|  | ||||
| function isDate(d) { | ||||
|     return isObject(d) && objectToString(d) === '[object Date]'; | ||||
|    return isObject(d) && objectToString(d) === "[object Date]"; | ||||
| } | ||||
|  | ||||
| function objectToString(o) { | ||||
| @ -155,35 +163,46 @@ function formatArray(ctx, value, recurseTimes, visibleKeys, keys) { | ||||
|    var output = []; | ||||
|    for (var i = 0, l = value.length; i < l; ++i) { | ||||
|       if (hasOwn(value, String(i))) { | ||||
|             output.push(formatProperty(ctx, value, recurseTimes, visibleKeys, | ||||
|                 String(i), true)); | ||||
|          output.push( | ||||
|             formatProperty( | ||||
|                ctx, | ||||
|                value, | ||||
|                recurseTimes, | ||||
|                visibleKeys, | ||||
|                String(i), | ||||
|                true | ||||
|             ) | ||||
|          ); | ||||
|       } else { | ||||
|             output.push(''); | ||||
|          output.push(""); | ||||
|       } | ||||
|    } | ||||
|    keys.forEach(function (key) { | ||||
|       if (!key.match(/^\d+$/)) { | ||||
|             output.push(formatProperty(ctx, value, recurseTimes, visibleKeys, | ||||
|                 key, true)); | ||||
|          output.push( | ||||
|             formatProperty(ctx, value, recurseTimes, visibleKeys, key, true) | ||||
|          ); | ||||
|       } | ||||
|    }); | ||||
|    return output; | ||||
| } | ||||
|  | ||||
| function formatError(value) { | ||||
|     return '[' + Error.prototype.toString.call(value) + ']'; | ||||
|    return "[" + Error.prototype.toString.call(value) + "]"; | ||||
| } | ||||
|  | ||||
| function formatValue(ctx, value, recurseTimes) { | ||||
|    // Provide a hook for user-specified inspect functions. | ||||
|    // Check that value is an object with an inspect function on it | ||||
|     if (ctx.customInspect && | ||||
|    if ( | ||||
|       ctx.customInspect && | ||||
|       value && | ||||
|       isFunction(value.inspect) && | ||||
|       // Filter out the util module, it's inspect function is special | ||||
|       value.inspect !== inspect && | ||||
|       // Also filter out any prototype objects using the circular check. | ||||
|         !(value.constructor && value.constructor.prototype === value)) { | ||||
|       !(value.constructor && value.constructor.prototype === value) | ||||
|    ) { | ||||
|       var ret = value.inspect(recurseTimes, ctx); | ||||
|       if (!isString(ret)) { | ||||
|          ret = formatValue(ctx, ret, recurseTimes); | ||||
| @ -211,55 +230,59 @@ function formatValue(ctx, value, recurseTimes) { | ||||
|  | ||||
|    // IE doesn't make error fields non-enumerable | ||||
|    // http://msdn.microsoft.com/en-us/library/ie/dww52sbt(v=vs.94).aspx | ||||
|     if (isError(value) | ||||
|         && (keys.indexOf('message') >= 0 || keys.indexOf('description') >= 0)) { | ||||
|    if ( | ||||
|       isError(value) && | ||||
|       (keys.indexOf("message") >= 0 || keys.indexOf("description") >= 0) | ||||
|    ) { | ||||
|       return formatError(value); | ||||
|    } | ||||
|  | ||||
|    // Some type of object without properties can be shortcutted. | ||||
|    if (keys.length === 0) { | ||||
|       if (isFunction(value)) { | ||||
|             var name = value.name ? ': ' + value.name : ''; | ||||
|             return ctx.stylize('[Function' + name + ']', 'special'); | ||||
|          var name = value.name ? ": " + value.name : ""; | ||||
|          return ctx.stylize("[Function" + name + "]", "special"); | ||||
|       } | ||||
|       if (isRegExp(value)) { | ||||
|             return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp'); | ||||
|          return ctx.stylize(RegExp.prototype.toString.call(value), "regexp"); | ||||
|       } | ||||
|       if (isDate(value)) { | ||||
|             return ctx.stylize(Date.prototype.toString.call(value), 'date'); | ||||
|          return ctx.stylize(Date.prototype.toString.call(value), "date"); | ||||
|       } | ||||
|       if (isError(value)) { | ||||
|          return formatError(value); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|     var base = '', array = false, braces = ['{', '}']; | ||||
|    var base = "", | ||||
|       array = false, | ||||
|       braces = ["{", "}"]; | ||||
|  | ||||
|    // Make Array say that they are Array | ||||
|    if (Array.isArray(value)) { | ||||
|       array = true; | ||||
|         braces = ['[', ']']; | ||||
|       braces = ["[", "]"]; | ||||
|    } | ||||
|  | ||||
|    // Make functions say that they are functions | ||||
|    if (isFunction(value)) { | ||||
|         var n = value.name ? ': ' + value.name : ''; | ||||
|         base = ' [Function' + n + ']'; | ||||
|       var n = value.name ? ": " + value.name : ""; | ||||
|       base = " [Function" + n + "]"; | ||||
|    } | ||||
|  | ||||
|    // Make RegExps say that they are RegExps | ||||
|    if (isRegExp(value)) { | ||||
|         base = ' ' + RegExp.prototype.toString.call(value); | ||||
|       base = " " + RegExp.prototype.toString.call(value); | ||||
|    } | ||||
|  | ||||
|    // Make dates with properties first say the date | ||||
|    if (isDate(value)) { | ||||
|         base = ' ' + Date.prototype.toUTCString.call(value); | ||||
|       base = " " + Date.prototype.toUTCString.call(value); | ||||
|    } | ||||
|  | ||||
|    // Make error with message first say the error | ||||
|    if (isError(value)) { | ||||
|         base = ' ' + formatError(value); | ||||
|       base = " " + formatError(value); | ||||
|    } | ||||
|  | ||||
|    if (keys.length === 0 && (!array || value.length == 0)) { | ||||
| @ -268,9 +291,9 @@ function formatValue(ctx, value, recurseTimes) { | ||||
|  | ||||
|    if (recurseTimes < 0) { | ||||
|       if (isRegExp(value)) { | ||||
|             return ctx.stylize(RegExp.prototype.toString.call(value), 'regexp'); | ||||
|          return ctx.stylize(RegExp.prototype.toString.call(value), "regexp"); | ||||
|       } else { | ||||
|             return ctx.stylize('[Object]', 'special'); | ||||
|          return ctx.stylize("[Object]", "special"); | ||||
|       } | ||||
|    } | ||||
|  | ||||
| @ -281,7 +304,14 @@ function formatValue(ctx, value, recurseTimes) { | ||||
|       output = formatArray(ctx, value, recurseTimes, visibleKeys, keys); | ||||
|    } else { | ||||
|       output = keys.map(function (key) { | ||||
|             return formatProperty(ctx, value, recurseTimes, visibleKeys, key, array); | ||||
|          return formatProperty( | ||||
|             ctx, | ||||
|             value, | ||||
|             recurseTimes, | ||||
|             visibleKeys, | ||||
|             key, | ||||
|             array | ||||
|          ); | ||||
|       }); | ||||
|    } | ||||
|  | ||||
| @ -311,17 +341,17 @@ function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) { | ||||
|    } | ||||
|    if (desc.get) { | ||||
|       if (desc.set) { | ||||
|             str = ctx.stylize('[Getter/Setter]', 'special'); | ||||
|          str = ctx.stylize("[Getter/Setter]", "special"); | ||||
|       } else { | ||||
|             str = ctx.stylize('[Getter]', 'special'); | ||||
|          str = ctx.stylize("[Getter]", "special"); | ||||
|       } | ||||
|    } else { | ||||
|       if (desc.set) { | ||||
|             str = ctx.stylize('[Setter]', 'special'); | ||||
|          str = ctx.stylize("[Setter]", "special"); | ||||
|       } | ||||
|    } | ||||
|    if (!hasOwn(visibleKeys, key)) { | ||||
|         name = '[' + key + ']'; | ||||
|       name = "[" + key + "]"; | ||||
|    } | ||||
|    if (!str) { | ||||
|       if (ctx.seen.indexOf(desc.value) < 0) { | ||||
| @ -330,76 +360,88 @@ function formatProperty(ctx, value, recurseTimes, visibleKeys, key, array) { | ||||
|          } else { | ||||
|             str = formatValue(ctx, desc.value, recurseTimes - 1); | ||||
|          } | ||||
|             if (str.indexOf('\n') > -1) { | ||||
|          if (str.indexOf("\n") > -1) { | ||||
|             if (array) { | ||||
|                     str = str.split('\n').map(function (line) { | ||||
|                         return '  ' + line; | ||||
|                     }).join('\n').substr(2); | ||||
|                str = str | ||||
|                   .split("\n") | ||||
|                   .map(function (line) { | ||||
|                      return "  " + line; | ||||
|                   }) | ||||
|                   .join("\n") | ||||
|                   .substr(2); | ||||
|             } else { | ||||
|                     str = '\n' + str.split('\n').map(function (line) { | ||||
|                         return '   ' + line; | ||||
|                     }).join('\n'); | ||||
|                str = | ||||
|                   "\n" + | ||||
|                   str | ||||
|                      .split("\n") | ||||
|                      .map(function (line) { | ||||
|                         return "   " + line; | ||||
|                      }) | ||||
|                      .join("\n"); | ||||
|             } | ||||
|          } | ||||
|       } else { | ||||
|             str = ctx.stylize('[Circular]', 'special'); | ||||
|          str = ctx.stylize("[Circular]", "special"); | ||||
|       } | ||||
|    } | ||||
|    if (isUndefined(name)) { | ||||
|       if (array && key.match(/^\d+$/)) { | ||||
|          return str; | ||||
|       } | ||||
|         name = JSON.stringify('' + key); | ||||
|       name = JSON.stringify("" + key); | ||||
|       if (name.match(/^"([a-zA-Z_][a-zA-Z_0-9]*)"$/)) { | ||||
|          name = name.substr(1, name.length - 2); | ||||
|             name = ctx.stylize(name, 'name'); | ||||
|          name = ctx.stylize(name, "name"); | ||||
|       } else { | ||||
|             name = name.replace(/'/g, "\\'") | ||||
|          name = name | ||||
|             .replace(/'/g, "\\'") | ||||
|             .replace(/\\"/g, '"') | ||||
|             .replace(/(^"|"$)/g, "'"); | ||||
|             name = ctx.stylize(name, 'string'); | ||||
|          name = ctx.stylize(name, "string"); | ||||
|       } | ||||
|    } | ||||
|  | ||||
|     return name + ': ' + str; | ||||
|    return name + ": " + str; | ||||
| } | ||||
|  | ||||
| function formatPrimitive(ctx, value) { | ||||
|     if (isUndefined(value)) | ||||
|         return ctx.stylize('undefined', 'undefined'); | ||||
|    if (isUndefined(value)) return ctx.stylize("undefined", "undefined"); | ||||
|    if (isString(value)) { | ||||
|         var simple = '\'' + JSON.stringify(value).replace(/^"|"$/g, '') | ||||
|       var simple = | ||||
|          "'" + | ||||
|          JSON.stringify(value) | ||||
|             .replace(/^"|"$/g, "") | ||||
|             .replace(/'/g, "\\'") | ||||
|             .replace(/\\"/g, '"') + '\''; | ||||
|         return ctx.stylize(simple, 'string'); | ||||
|             .replace(/\\"/g, '"') + | ||||
|          "'"; | ||||
|       return ctx.stylize(simple, "string"); | ||||
|    } | ||||
|     if (isNumber(value)) | ||||
|         return ctx.stylize('' + value, 'number'); | ||||
|     if (isBoolean(value)) | ||||
|         return ctx.stylize('' + value, 'boolean'); | ||||
|    if (isNumber(value)) return ctx.stylize("" + value, "number"); | ||||
|    if (isBoolean(value)) return ctx.stylize("" + value, "boolean"); | ||||
|    // For some reason typeof null is "object", so special case here. | ||||
|     if (isNull(value)) | ||||
|         return ctx.stylize('null', 'null'); | ||||
|    if (isNull(value)) return ctx.stylize("null", "null"); | ||||
| } | ||||
|  | ||||
| function reduceToSingleString(output, base, braces) { | ||||
|    var numLinesEst = 0; | ||||
|    var length = output.reduce(function (prev, cur) { | ||||
|       numLinesEst++; | ||||
|         if (cur.indexOf('\n') >= 0) numLinesEst++; | ||||
|         return prev + cur.replace(/\u001b\[\d\d?m/g, '').length + 1; | ||||
|       if (cur.indexOf("\n") >= 0) numLinesEst++; | ||||
|       return prev + cur.replace(/\u001b\[\d\d?m/g, "").length + 1; | ||||
|    }, 0); | ||||
|  | ||||
|    if (length > 60) { | ||||
|         return braces[0] + | ||||
|             (base === '' ? '' : base + '\n ') + | ||||
|             ' ' + | ||||
|             output.join(',\n  ') + | ||||
|             ' ' + | ||||
|             braces[1]; | ||||
|       return ( | ||||
|          braces[0] + | ||||
|          (base === "" ? "" : base + "\n ") + | ||||
|          " " + | ||||
|          output.join(",\n  ") + | ||||
|          " " + | ||||
|          braces[1] | ||||
|       ); | ||||
|    } | ||||
|  | ||||
|     return braces[0] + base + ' ' + output.join(', ') + ' ' + braces[1]; | ||||
|    return braces[0] + base + " " + output.join(", ") + " " + braces[1]; | ||||
| } | ||||
|  | ||||
| function _extend<T extends Y, Y>(origin: T, add: Y) { | ||||
|  | ||||
							
								
								
									
										82
									
								
								src/test.ts
									
									
									
									
									
								
							
							
						
						
									
										82
									
								
								src/test.ts
									
									
									
									
									
								
							| @ -1,34 +1,72 @@ | ||||
| import { Logging, LoggingBase, Colors, withColor } from "."; | ||||
| import Logging from "./index.js"; | ||||
|  | ||||
| Logging.log("test") | ||||
| 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( | ||||
|    "\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) | ||||
| let err = new Error(); | ||||
| if (typeof err.stack !== "string") console.log("Stacktrace invalid", err.stack); | ||||
|  | ||||
| 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!"); | ||||
|  | ||||
| let cus2 = new LoggingBase("test2"); | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| let cus2 = new LoggingBase({ name: "test2" }); | ||||
| cus2.log("Hello from custom Logger 2"); | ||||
|  | ||||
| let cus22 = new LoggingBase("test2"); | ||||
| cus22.log("Hello from custom Logger 22") | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| cus22.log("Hello from custom Logger 22") | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| cus22.log("Hello from custom Logger 22") | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| cus22.log("Hello from custom Logger 22") | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| cus22.log("Hello from custom Logger 22") | ||||
| cus2.log("Hello from custom Logger 2") | ||||
| 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"); | ||||
| cus2.log("Hello from custom Logger 2"); | ||||
| cus22.log("Hello from custom Logger 22"); | ||||
| cus2.log("Hello from custom Logger 2"); | ||||
| cus22.log("Hello from custom Logger 22"); | ||||
| cus2.log("Hello from custom Logger 2"); | ||||
| cus22.log("Hello from custom Logger 22"); | ||||
| cus2.log("Hello from custom Logger 2"); | ||||
|  | ||||
| Logging.debug("Only Errors should appear:"); | ||||
| Logging.logLevel = LoggingTypes.Error; | ||||
|  | ||||
| Logging.debug("This should not be there 1"); | ||||
| Logging.log("This should not be there 2"); | ||||
| Logging.warn("This should not be there 3"); | ||||
| Logging.warning("This should not be there 4"); | ||||
|  | ||||
| Logging.error("This should be there 1"); | ||||
| Logging.error("This should be there 2"); | ||||
|  | ||||
| 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"); | ||||
|  | ||||
							
								
								
									
										246
									
								
								src/types.ts
									
									
									
									
									
								
							
							
						
						
									
										246
									
								
								src/types.ts
									
									
									
									
									
								
							| @ -1,13 +1,10 @@ | ||||
| import { ObservableInterface } from "@hibas123/utils"; | ||||
|  | ||||
| export enum LoggingTypes { | ||||
|    Debug, | ||||
|    Log, | ||||
|     Warning, | ||||
|    Warn, | ||||
|    Error, | ||||
|     Debug | ||||
| } | ||||
|  | ||||
|  | ||||
| export const TerminalFormats = { | ||||
|    Reset: "\x1b[0m", | ||||
|    Bold: "\x1b[1m", | ||||
| @ -32,18 +29,10 @@ export const TerminalFormats = { | ||||
|    BgBlue: "\x1b[44m", | ||||
|    BgMagenta: "\x1b[45m", | ||||
|    BgCyan: "\x1b[46m", | ||||
|     BgWhite: "\x1b[47m" | ||||
| } | ||||
|    BgWhite: "\x1b[47m", | ||||
| }; | ||||
|  | ||||
|  | ||||
| export enum FormatTypes { | ||||
|     COLOR, | ||||
|     BOLD, | ||||
|     UNDERSCORE, | ||||
|     BLINK | ||||
| } | ||||
|  | ||||
| export enum Colors { | ||||
| export enum IColors { | ||||
|    NONE, | ||||
|    RED, | ||||
|    GREEN, | ||||
| @ -51,69 +40,218 @@ export enum Colors { | ||||
|    BLUE, | ||||
|    MAGENTA, | ||||
|    CYAN, | ||||
|     WHITE | ||||
|    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; | ||||
|       } | ||||
|    } | ||||
|  | ||||
| const boldFormat = { | ||||
|     type: FormatTypes.BOLD | ||||
|    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 { | ||||
|    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 { | ||||
|     init(observable: ObservableInterface<Message>, name?: string): void | Promise<void>; | ||||
|    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(): 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. | ||||
|     * | ||||
|     * WARNING: The adapter might be reinitialised, when it is added to a new Logging instance | ||||
|     */ | ||||
|    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
	