Compare commits
	
		
			49 Commits
		
	
	
		
			ce8ecbe8c9
			...
			4.0.1
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 453128fc4f | |||
| a399225c43 | |||
| 08fabf5bf8 | |||
| 6c65d2c83d | |||
| ee3123f400 | |||
| 95ef923844 | |||
| ccb5aa023f | |||
| c372016397 | |||
| ca8dffecff | |||
| d39e13dfe1 | |||
| 0742490527 | |||
| c58d75129d | |||
| 249f701cb7 | |||
| f01a4ffb21 | |||
| 43b94c5c75 | |||
| 9600b46699 | |||
| b3f6a6c3f2 | |||
| 090aa4629b | |||
| 558ddad800 | |||
| 6f002456f6 | |||
| a8a388997f | |||
| 2a55549199 | |||
| c7c968753f | |||
| c5098934a1 | |||
| 922af328a3 | |||
| b76d0083b6 | |||
| 57f1b07944 | |||
| a6a5eeded5 | |||
| 2735bcba35 | |||
| 4c656420ad | |||
| cc73129373 | |||
| 9c454b403f | |||
| 5bbf0f74f5 | |||
| 8bf0d4b798 | |||
| 0dbd8e9c40 | |||
| 27da76c1b0 | |||
| 096c5910c3 | |||
| 2aaee1be89 | |||
| 0b75f8ddf8 | |||
| 03cc58d3e1 | |||
| 72f06a88d6 | |||
| 9417264850 | |||
| 58ff2fd2ea | |||
| dec35001e3 | |||
| fbb55fa158 | |||
| 331169c925 | |||
| 0f3dc07dff | |||
| 7f72d28d22 | |||
| 37b00fd772 | 
| @ -1,3 +1,4 @@ | |||||||
|  | [*] | ||||||
| charset = utf-8 | charset = utf-8 | ||||||
| indent_style = space | indent_style = space | ||||||
| indent_size = 3 | indent_size = 3 | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | /.yarn/**            linguist-vendored | ||||||
|  | /.yarn/releases/*    binary | ||||||
|  | /.yarn/plugins/**/*  binary | ||||||
|  | /.pnp.*              binary linguist-generated | ||||||
							
								
								
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										5
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -1,3 +1,8 @@ | |||||||
| node_modules/ | node_modules/ | ||||||
| logs/ | logs/ | ||||||
| yarn.lock | yarn.lock | ||||||
|  | out/ | ||||||
|  | esm/ | ||||||
|  | .history/ | ||||||
|  | .yarn/cache | ||||||
|  | .yarn/install-state.gz | ||||||
							
								
								
									
										541
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										541
									
								
								.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										874
									
								
								.yarn/releases/yarn-3.6.4.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
							
						
						
									
										874
									
								
								.yarn/releases/yarn-3.6.4.cjs
									
									
									
									
										vendored
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										11
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								.yarnrc.yml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | nodeLinker: node-modules | ||||||
|  |  | ||||||
|  | npmScopes: | ||||||
|  |    "hibas123": | ||||||
|  |       npmRegistryServer: "https://git.hibas.dev/api/packages/hibas123/npm/" | ||||||
|  |  | ||||||
|  | plugins: | ||||||
|  |    - path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs | ||||||
|  |      spec: "@yarnpkg/plugin-interactive-tools" | ||||||
|  |  | ||||||
|  | yarnPath: .yarn/releases/yarn-3.6.4.cjs | ||||||
							
								
								
									
										69
									
								
								out/index.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										69
									
								
								out/index.d.ts
									
									
									
									
										vendored
									
									
								
							| @ -1,69 +0,0 @@ | |||||||
| /// <reference types="node" /> |  | ||||||
| import { EventEmitter } from "events"; |  | ||||||
| export declare const Colors: { |  | ||||||
|     Reset: string; |  | ||||||
|     Bright: string; |  | ||||||
|     Dim: string; |  | ||||||
|     Underscore: string; |  | ||||||
|     Blink: string; |  | ||||||
|     Reverse: string; |  | ||||||
|     Hidden: string; |  | ||||||
|     FgBlack: string; |  | ||||||
|     FgRed: string; |  | ||||||
|     FgGreen: string; |  | ||||||
|     FgYellow: string; |  | ||||||
|     FgBlue: string; |  | ||||||
|     FgMagenta: string; |  | ||||||
|     FgCyan: string; |  | ||||||
|     FgWhite: string; |  | ||||||
|     BgBlack: string; |  | ||||||
|     BgRed: string; |  | ||||||
|     BgGreen: string; |  | ||||||
|     BgYellow: string; |  | ||||||
|     BgBlue: string; |  | ||||||
|     BgMagenta: string; |  | ||||||
|     BgCyan: string; |  | ||||||
|     BgWhite: string; |  | ||||||
| }; |  | ||||||
| export interface LoggingBaseOptions { |  | ||||||
|     /** |  | ||||||
|      * Name will be prefixed on Console output and added to logfiles, if not specified here |  | ||||||
|      */ |  | ||||||
|     name: string; |  | ||||||
|     /** |  | ||||||
|      * Filename/path of the logfile. Skip if generated with name |  | ||||||
|      */ |  | ||||||
|     logfile: string; |  | ||||||
|     /** |  | ||||||
|      * Filename/path of the logfile. Skip if generated with name |  | ||||||
|      */ |  | ||||||
|     errorfile: string; |  | ||||||
|     /** |  | ||||||
|      * Prints output to console |  | ||||||
|      */ |  | ||||||
|     console_out: boolean; |  | ||||||
| } |  | ||||||
| export declare class LoggingBase { |  | ||||||
|     private config; |  | ||||||
|     private logFile; |  | ||||||
|     private errorFile; |  | ||||||
|     constructor(options?: Partial<LoggingBaseOptions> | string); |  | ||||||
|     console_out: boolean; |  | ||||||
|     waitForSetup(): Promise<any[]>; |  | ||||||
|     events: EventEmitter; |  | ||||||
|     debug(...message: any[]): void; |  | ||||||
|     log(...message: any[]): void; |  | ||||||
|     warning(...message: any[]): void; |  | ||||||
|     logWithCustomColors(type: LoggingTypes, colors: string, ...message: any[]): void; |  | ||||||
|     error(error: Error | string): void; |  | ||||||
|     errorMessage(...message: any[]): void; |  | ||||||
|     private message(type, message, customColors?, caller?); |  | ||||||
| } |  | ||||||
| export declare let Logging: LoggingBase; |  | ||||||
| export default Logging; |  | ||||||
| export declare enum LoggingTypes { |  | ||||||
|     Log = 0, |  | ||||||
|     Warning = 1, |  | ||||||
|     Error = 2, |  | ||||||
|     Debug = 3, |  | ||||||
| } |  | ||||||
							
								
								
									
										397
									
								
								out/index.js
									
									
									
									
									
								
							
							
						
						
									
										397
									
								
								out/index.js
									
									
									
									
									
								
							| @ -1,397 +0,0 @@ | |||||||
| "use strict"; |  | ||||||
| Object.defineProperty(exports, "__esModule", { value: true }); |  | ||||||
| const util = require("util"); |  | ||||||
| const fs = require("fs"); |  | ||||||
| const events_1 = require("events"); |  | ||||||
| const path = require("path"); |  | ||||||
| const lock_1 = require("./lock"); |  | ||||||
| exports.Colors = { |  | ||||||
|     Reset: "\x1b[0m", |  | ||||||
|     Bright: "\x1b[1m", |  | ||||||
|     Dim: "\x1b[2m", |  | ||||||
|     Underscore: "\x1b[4m", |  | ||||||
|     Blink: "\x1b[5m", |  | ||||||
|     Reverse: "\x1b[7m", |  | ||||||
|     Hidden: "\x1b[8m", |  | ||||||
|     FgBlack: "\x1b[30m", |  | ||||||
|     FgRed: "\x1b[31m", |  | ||||||
|     FgGreen: "\x1b[32m", |  | ||||||
|     FgYellow: "\x1b[33m", |  | ||||||
|     FgBlue: "\x1b[34m", |  | ||||||
|     FgMagenta: "\x1b[35m", |  | ||||||
|     FgCyan: "\x1b[36m", |  | ||||||
|     FgWhite: "\x1b[37m", |  | ||||||
|     BgBlack: "\x1b[40m", |  | ||||||
|     BgRed: "\x1b[41m", |  | ||||||
|     BgGreen: "\x1b[42m", |  | ||||||
|     BgYellow: "\x1b[43m", |  | ||||||
|     BgBlue: "\x1b[44m", |  | ||||||
|     BgMagenta: "\x1b[45m", |  | ||||||
|     BgCyan: "\x1b[46m", |  | ||||||
|     BgWhite: "\x1b[47m" |  | ||||||
| }; |  | ||||||
| const maxFileSize = 500000000; |  | ||||||
| const OriginalErrorStackFunction = Error.prototype.prepareStackTrace; |  | ||||||
| class LoggingFiles { |  | ||||||
|     constructor(file) { |  | ||||||
|         this.size = 0; |  | ||||||
|         this.stream = undefined; |  | ||||||
|         this.lock = new lock_1.default(); |  | ||||||
|         this.queue = []; |  | ||||||
|         this.file = path.resolve(file); |  | ||||||
|         this.init(); |  | ||||||
|     } |  | ||||||
|     static getFile(filename) { |  | ||||||
|         filename = path.resolve(filename); |  | ||||||
|         let file = this.files.find(e => e.file === filename); |  | ||||||
|         if (!file) { |  | ||||||
|             file = new LoggingFiles(filename); |  | ||||||
|             this.files.push(file); |  | ||||||
|         } |  | ||||||
|         return file; |  | ||||||
|     } |  | ||||||
|     async awaitinit() { |  | ||||||
|         (await this.lock.getLock()).release(); |  | ||||||
|     } |  | ||||||
|     async init() { |  | ||||||
|         let lock = await this.lock.getLock(); |  | ||||||
|         await this.initializeFile(); |  | ||||||
|         lock.release(); |  | ||||||
|         this.checkQueue(); |  | ||||||
|     } |  | ||||||
|     async initializeFile(new_file = false) { |  | ||||||
|         try { |  | ||||||
|             if (this.stream) { |  | ||||||
|                 this.stream.close(); |  | ||||||
|             } |  | ||||||
|             const folder = path.dirname(this.file); |  | ||||||
|             if (folder) { |  | ||||||
|                 if (!await fsExists(folder)) { |  | ||||||
|                     await fsMkDir(folder).catch(() => { }); //Could happen, if two seperate instances want to create the same folder so ignoring |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             let size = 0; |  | ||||||
|             if (await fsExists(this.file)) { |  | ||||||
|                 let stats = await fsStat(this.file); |  | ||||||
|                 if (new_file || stats.size >= maxFileSize) { |  | ||||||
|                     if (await fsExists(this.file + ".old")) |  | ||||||
|                         await fsUnlink(this.file + ".old"); |  | ||||||
|                     await fsMove(this.file, this.file + ".old"); |  | ||||||
|                 } |  | ||||||
|                 else { |  | ||||||
|                     size = stats.size; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             this.stream = fs.createWriteStream(this.file, { flags: "a" }); |  | ||||||
|             this.size = size; |  | ||||||
|         } |  | ||||||
|         catch (e) { |  | ||||||
|             console.log(e); |  | ||||||
|             //ToDo is this the right behavior? |  | ||||||
|             process.exit(1); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     async checkQueue() { |  | ||||||
|         if (this.lock.locked) |  | ||||||
|             return; |  | ||||||
|         let lock = await this.lock.getLock(); |  | ||||||
|         let msg; |  | ||||||
|         while (msg = this.queue.shift()) { |  | ||||||
|             await this.write_to_file(msg); |  | ||||||
|         } |  | ||||||
|         lock.release(); |  | ||||||
|     } |  | ||||||
|     async write_to_file(data) { |  | ||||||
|         try { |  | ||||||
|             if (data.byteLength < maxFileSize && this.size + data.byteLength > maxFileSize) { |  | ||||||
|                 await this.initializeFile(true); |  | ||||||
|             } |  | ||||||
|             this.size += data.byteLength; |  | ||||||
|             this.stream.write(data); |  | ||||||
|         } |  | ||||||
|         catch (err) { |  | ||||||
|             console.error(err); |  | ||||||
|             this.initializeFile(false); |  | ||||||
|             this.write_to_file(data); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     write(data) { |  | ||||||
|         this.queue.push(data); |  | ||||||
|         this.checkQueue(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| LoggingFiles.files = []; |  | ||||||
| class LoggingBase { |  | ||||||
|     constructor(options) { |  | ||||||
|         this.events = new events_1.EventEmitter(); |  | ||||||
|         let opt; |  | ||||||
|         if (!options) |  | ||||||
|             opt = {}; |  | ||||||
|         else if (typeof options === "string") { |  | ||||||
|             opt = { name: options }; |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             opt = options; |  | ||||||
|         } |  | ||||||
|         if (opt.name) { |  | ||||||
|             if (opt.logfile === undefined) { |  | ||||||
|                 opt.logfile = `./logs/all.${opt.name}.log`; |  | ||||||
|             } |  | ||||||
|             if (opt.errorfile === undefined) { |  | ||||||
|                 opt.errorfile = `./logs/error.${opt.name}.log`; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         this.config = Object.assign({ |  | ||||||
|             name: undefined, |  | ||||||
|             console_out: true, |  | ||||||
|             logfile: "./logs/all.log", |  | ||||||
|             errorfile: "./logs/error.log" |  | ||||||
|         }, opt); |  | ||||||
|         for (let key in this) { |  | ||||||
|             if (typeof this[key] === "function") |  | ||||||
|                 this[key] = this[key].bind(this); |  | ||||||
|         } |  | ||||||
|         if (this.config.logfile) { |  | ||||||
|             this.logFile = LoggingFiles.getFile(this.config.logfile); |  | ||||||
|         } |  | ||||||
|         if (this.config.errorfile) { |  | ||||||
|             this.errorFile = LoggingFiles.getFile(this.config.errorfile); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     get console_out() { |  | ||||||
|         return this.config.console_out; |  | ||||||
|     } |  | ||||||
|     set console_out(value) { |  | ||||||
|         this.config.console_out = value; |  | ||||||
|     } |  | ||||||
|     waitForSetup() { |  | ||||||
|         let w = []; |  | ||||||
|         if (this.logFile) |  | ||||||
|             w.push(this.logFile.awaitinit()); |  | ||||||
|         if (this.errorFile) |  | ||||||
|             w.push(this.errorFile.awaitinit()); |  | ||||||
|         return Promise.all(w); |  | ||||||
|     } |  | ||||||
|     debug(...message) { |  | ||||||
|         this.message(LoggingTypes.Debug, message); |  | ||||||
|     } |  | ||||||
|     log(...message) { |  | ||||||
|         this.message(LoggingTypes.Log, message); |  | ||||||
|     } |  | ||||||
|     warning(...message) { |  | ||||||
|         this.message(LoggingTypes.Warning, message); |  | ||||||
|     } |  | ||||||
|     logWithCustomColors(type, colors, ...message) { |  | ||||||
|         this.message(type, message, colors); |  | ||||||
|     } |  | ||||||
|     error(error) { |  | ||||||
|         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]); |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             this.message(LoggingTypes.Error, [error.message, "\n", error.stack], undefined, getCallerFromExisting(error)); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     errorMessage(...message) { |  | ||||||
|         this.message(LoggingTypes.Error, message); |  | ||||||
|     } |  | ||||||
|     message(type, message, customColors, caller) { |  | ||||||
|         var consoleLogFormat = exports.Colors.Reset; |  | ||||||
|         if (!customColors) { |  | ||||||
|             switch (type) { |  | ||||||
|                 case LoggingTypes.Log: |  | ||||||
|                     //m += FgWhite + BgBlack; |  | ||||||
|                     break; |  | ||||||
|                 case LoggingTypes.Error: |  | ||||||
|                     consoleLogFormat += exports.Colors.FgRed; //FgWhite + BgRed + FgWhite; |  | ||||||
|                     break; |  | ||||||
|                 case LoggingTypes.Debug: |  | ||||||
|                     consoleLogFormat += exports.Colors.FgCyan; |  | ||||||
|                     break; |  | ||||||
|                 case LoggingTypes.Warning: |  | ||||||
|                     consoleLogFormat += exports.Colors.FgYellow; |  | ||||||
|                     break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             consoleLogFormat += customColors; |  | ||||||
|         } |  | ||||||
|         var mb = ""; |  | ||||||
|         if (typeof message === "string") { |  | ||||||
|             mb = message; |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             message.forEach((e, i) => { |  | ||||||
|                 if (typeof e !== "string") |  | ||||||
|                     e = util.inspect(e, false, null); |  | ||||||
|                 if (e.endsWith("\n") || i === message.length - 1) { |  | ||||||
|                     mb += e; |  | ||||||
|                 } |  | ||||||
|                 else { |  | ||||||
|                     mb += e + " "; |  | ||||||
|                 } |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|         let file = caller || getCallerFile(); |  | ||||||
|         let date = new Date().toISOString().replace(/T/, ' ').replace(/\..+/, ''); |  | ||||||
|         let prefix = `[${LoggingTypes[type]}][${file.file}:${file.line}][${date}]: `; |  | ||||||
|         let message_lines = mb.split("\n").map(line => prefix + line); |  | ||||||
|         if (this.config.console_out) { |  | ||||||
|             let name = ""; |  | ||||||
|             if (this.config.name) |  | ||||||
|                 name = `[${this.config.name}]`; |  | ||||||
|             message_lines.forEach(line => console.log(consoleLogFormat + name + line + exports.Colors.Reset)); |  | ||||||
|         } |  | ||||||
|         let m = message_lines.join("\n"); |  | ||||||
|         m = m.replace(/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g, ""); |  | ||||||
|         let index = m.indexOf("\x1b"); |  | ||||||
|         while (index >= 0) { |  | ||||||
|             m = m.substring(0, index) + m.substring(index + 5, m.length); |  | ||||||
|             index = m.indexOf("\x1b"); |  | ||||||
|         } |  | ||||||
|         let data = Buffer.from(m + "\n"); |  | ||||||
|         if (type === LoggingTypes.Error && this.errorFile) { |  | ||||||
|             this.errorFile.write(data); |  | ||||||
|         } |  | ||||||
|         if (this.logFile) { |  | ||||||
|             this.logFile.write(data); |  | ||||||
|         } |  | ||||||
|         this.events.emit("message", { type: type, message: m, module: this.config.name }); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| exports.LoggingBase = LoggingBase; |  | ||||||
| exports.Logging = undefined; |  | ||||||
| if (process.env.LOGGING_NO_DEFAULT !== "true") { |  | ||||||
|     exports.Logging = new LoggingBase(); |  | ||||||
| } |  | ||||||
| exports.default = exports.Logging; |  | ||||||
| function fsUnlink(path) { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|         fs.unlink(path, (err) => { |  | ||||||
|             if (err) |  | ||||||
|                 reject(err); |  | ||||||
|             else |  | ||||||
|                 resolve(); |  | ||||||
|         }); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| function fsStat(path) { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|         fs.stat(path, (err, stats) => { |  | ||||||
|             if (err) |  | ||||||
|                 reject(err); |  | ||||||
|             else |  | ||||||
|                 resolve(stats); |  | ||||||
|         }); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| function fsMove(oldPath, newPath) { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|         let callback = (err) => { |  | ||||||
|             if (err) |  | ||||||
|                 reject(err); |  | ||||||
|             else |  | ||||||
|                 resolve(); |  | ||||||
|         }; |  | ||||||
|         fs.rename(oldPath, newPath, function (err) { |  | ||||||
|             if (err) { |  | ||||||
|                 if (err.code === 'EXDEV') { |  | ||||||
|                     copy(); |  | ||||||
|                 } |  | ||||||
|                 else { |  | ||||||
|                     callback(err); |  | ||||||
|                 } |  | ||||||
|                 return; |  | ||||||
|             } |  | ||||||
|             callback(); |  | ||||||
|         }); |  | ||||||
|         function copy() { |  | ||||||
|             fs.copyFile(oldPath, newPath, (err) => { |  | ||||||
|                 if (err) |  | ||||||
|                     callback(err); |  | ||||||
|                 else |  | ||||||
|                     fs.unlink(oldPath, callback); |  | ||||||
|             }); |  | ||||||
|             // var readStream = fs.createReadStream(oldPath); |  | ||||||
|             // var writeStream = fs.createWriteStream(newPath); |  | ||||||
|             // readStream.on('error', callback); |  | ||||||
|             // writeStream.on('error', callback); |  | ||||||
|             // readStream.on('close', function () { |  | ||||||
|             //    fs.unlink(oldPath, callback); |  | ||||||
|             // }); |  | ||||||
|             // readStream.pipe(writeStream); |  | ||||||
|         } |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| function fsExists(path) { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|         fs.exists(path, resolve); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| function fsMkDir(path) { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|         fs.mkdir(path, (err) => err ? reject(err) : resolve()); |  | ||||||
|     }); |  | ||||||
| } |  | ||||||
| function getStack() { |  | ||||||
|     // Save original Error.prepareStackTrace |  | ||||||
|     let origPrepareStackTrace = Error.prepareStackTrace; |  | ||||||
|     // Override with function that just returns `stack` |  | ||||||
|     Error.prepareStackTrace = function (_, stack) { |  | ||||||
|         return stack; |  | ||||||
|     }; |  | ||||||
|     // Create a new `Error`, which automatically gets `stack` |  | ||||||
|     let err = new Error(); |  | ||||||
|     // Evaluate `err.stack`, which calls our new `Error.prepareStackTrace` |  | ||||||
|     let stack = err.stack; |  | ||||||
|     // Restore original `Error.prepareStackTrace` |  | ||||||
|     Error.prepareStackTrace = origPrepareStackTrace; |  | ||||||
|     // Remove superfluous function call on stack |  | ||||||
|     stack.shift(); // getStack --> Error |  | ||||||
|     return stack; |  | ||||||
| } |  | ||||||
| function getCallerFile() { |  | ||||||
|     try { |  | ||||||
|         let stack = getStack(); |  | ||||||
|         let current_file = stack.shift().getFileName(); |  | ||||||
|         while (stack.length) { |  | ||||||
|             let caller_file = stack.shift(); |  | ||||||
|             const util = require("util"); |  | ||||||
|             if (current_file !== caller_file.getFileName()) |  | ||||||
|                 return { |  | ||||||
|                     file: path.basename(caller_file.getFileName()), |  | ||||||
|                     line: caller_file.getLineNumber() |  | ||||||
|                 }; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     catch (err) { } |  | ||||||
|     return { file: undefined, line: 0 }; |  | ||||||
| } |  | ||||||
| function getCallerFromExisting(err) { |  | ||||||
|     let lines = err.stack.split("\n"); |  | ||||||
|     let current = path.basename(__filename); |  | ||||||
|     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); |  | ||||||
|         if (matches && matches.length > 0) { |  | ||||||
|             let [f, line] = matches[0].split(":"); |  | ||||||
|             if (f != current) { |  | ||||||
|                 return { |  | ||||||
|                     file: f, line: Number(line) |  | ||||||
|                 }; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| var LoggingTypes; |  | ||||||
| (function (LoggingTypes) { |  | ||||||
|     LoggingTypes[LoggingTypes["Log"] = 0] = "Log"; |  | ||||||
|     LoggingTypes[LoggingTypes["Warning"] = 1] = "Warning"; |  | ||||||
|     LoggingTypes[LoggingTypes["Error"] = 2] = "Error"; |  | ||||||
|     LoggingTypes[LoggingTypes["Debug"] = 3] = "Debug"; |  | ||||||
| })(LoggingTypes = exports.LoggingTypes || (exports.LoggingTypes = {})); |  | ||||||
| //# sourceMappingURL=index.js.map |  | ||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										12
									
								
								out/lock.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								out/lock.d.ts
									
									
									
									
										vendored
									
									
								
							| @ -1,12 +0,0 @@ | |||||||
| export declare type Release = { |  | ||||||
|     release: () => void; |  | ||||||
| }; |  | ||||||
| export default class Lock { |  | ||||||
|     private _locked; |  | ||||||
|     readonly locked: boolean; |  | ||||||
|     private toCome; |  | ||||||
|     constructor(); |  | ||||||
|     getLock(): Promise<Release>; |  | ||||||
|     private lock(); |  | ||||||
|     private release(); |  | ||||||
| } |  | ||||||
							
								
								
									
										37
									
								
								out/lock.js
									
									
									
									
									
								
							
							
						
						
									
										37
									
								
								out/lock.js
									
									
									
									
									
								
							| @ -1,37 +0,0 @@ | |||||||
| "use strict"; |  | ||||||
| Object.defineProperty(exports, "__esModule", { value: true }); |  | ||||||
| class Lock { |  | ||||||
|     constructor() { |  | ||||||
|         this._locked = false; |  | ||||||
|         this.toCome = []; |  | ||||||
|         this.release = this.release.bind(this); |  | ||||||
|     } |  | ||||||
|     get locked() { |  | ||||||
|         return this._locked; |  | ||||||
|     } |  | ||||||
|     async getLock() { |  | ||||||
|         if (!this._locked) |  | ||||||
|             return { release: this.lock() }; |  | ||||||
|         else { |  | ||||||
|             return new Promise((resolve) => { |  | ||||||
|                 this.toCome.push(() => { |  | ||||||
|                     resolve({ release: this.lock() }); |  | ||||||
|                 }); |  | ||||||
|             }); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|     lock() { |  | ||||||
|         this._locked = true; |  | ||||||
|         return this.release; |  | ||||||
|     } |  | ||||||
|     async release() { |  | ||||||
|         if (this.toCome.length > 0) { |  | ||||||
|             this.toCome.shift()(); |  | ||||||
|         } |  | ||||||
|         else { |  | ||||||
|             this._locked = false; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| exports.default = Lock; |  | ||||||
| //# sourceMappingURL=lock.js.map |  | ||||||
| @ -1 +0,0 @@ | |||||||
| {"version":3,"file":"lock.js","sourceRoot":"","sources":["../src/lock.ts"],"names":[],"mappings":";;AACA;IAOG;QANQ,YAAO,GAAY,KAAK,CAAC;QAIzB,WAAM,GAAmB,EAAE,CAAC;QAGjC,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAC1C,CAAC;IAPD,IAAI,MAAM;QACP,OAAO,IAAI,CAAC,OAAO,CAAC;IACvB,CAAC;IAOD,KAAK,CAAC,OAAO;QACV,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO,EAAE,OAAO,EAAE,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC;aAC9C;YACF,OAAO,IAAI,OAAO,CAAU,CAAC,OAAO,EAAE,EAAE;gBACrC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,EAAE;oBACnB,OAAO,CAAC,EAAE,OAAO,EAAE,IAAI,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;gBACrC,CAAC,CAAC,CAAA;YACL,CAAC,CAAC,CAAA;SACJ;IACJ,CAAC;IAEO,IAAI;QACT,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACpB,OAAO,IAAI,CAAC,OAAO,CAAC;IACvB,CAAC;IAEO,KAAK,CAAC,OAAO;QAClB,IAAI,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE;YACzB,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,EAAE,CAAC;SACxB;aAAM;YACJ,IAAI,CAAC,OAAO,GAAG,KAAK,CAAC;SACvB;IACJ,CAAC;CACH;AAlCD,uBAkCC"} |  | ||||||
							
								
								
									
										1
									
								
								out/test.d.ts
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								out/test.d.ts
									
									
									
									
										vendored
									
									
								
							| @ -1 +0,0 @@ | |||||||
| export {}; |  | ||||||
							
								
								
									
										50
									
								
								out/test.js
									
									
									
									
									
								
							
							
						
						
									
										50
									
								
								out/test.js
									
									
									
									
									
								
							| @ -1,50 +0,0 @@ | |||||||
| "use strict"; |  | ||||||
| Object.defineProperty(exports, "__esModule", { value: true }); |  | ||||||
| const index_1 = require("./index"); |  | ||||||
| const crypto_1 = require("crypto"); |  | ||||||
| index_1.Logging.log("test"); |  | ||||||
| index_1.Logging.log("i", "am", { a: "an" }, 1000); |  | ||||||
| index_1.Logging.error(new Error("fehler 001")); |  | ||||||
| index_1.Logging.debug("Some Debug infos"); |  | ||||||
| index_1.Logging.errorMessage("i", "am", "an", "error"); |  | ||||||
| index_1.Logging.log("\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m\x1b[31m TEST \x1b[31m\x1b[31m\x1b[31m"); |  | ||||||
| let err = new Error(); |  | ||||||
| if (typeof err.stack !== "string") |  | ||||||
|     console.log("Stacktrace invalid", err.stack); |  | ||||||
| let cus = new index_1.LoggingBase({ name: "test" }); |  | ||||||
| cus.log("Hello from custom Logger"); |  | ||||||
| let cus2 = new index_1.LoggingBase("test2"); |  | ||||||
| cus2.log("Hello from custom Logger 2"); |  | ||||||
| let cus22 = new index_1.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"); |  | ||||||
| index_1.Logging.console_out = false; |  | ||||||
| async function benchmark(count, message_size) { |  | ||||||
|     await index_1.Logging.waitForSetup(); |  | ||||||
|     const randData = crypto_1.randomBytes(message_size).toString("hex"); |  | ||||||
|     const t = process.hrtime(); |  | ||||||
|     for (let i = 0; i < count; i++) { |  | ||||||
|         index_1.Logging.log(randData); |  | ||||||
|     } |  | ||||||
|     const diff = process.hrtime(t); |  | ||||||
|     const NS_PER_SEC = 1e9; |  | ||||||
|     await index_1.Logging.waitForSetup(); |  | ||||||
|     const ns = diff[0] * NS_PER_SEC + diff[1]; |  | ||||||
|     console.log(`Benchmark took ${ns / 1000000}ms for ${count} messages with a size of ${message_size} characters`); |  | ||||||
|     console.log(`This is equal to ${(ns / 1000000) / count} ms per message`); |  | ||||||
| } |  | ||||||
| index_1.Logging.waitForSetup().then(async () => { |  | ||||||
|     console.log("Large data benchmark:"); |  | ||||||
|     await benchmark(7000, 50000); |  | ||||||
|     console.log("Realdata data benchmark:"); |  | ||||||
|     await benchmark(100000, 100); |  | ||||||
| }); |  | ||||||
| //# sourceMappingURL=test.js.map |  | ||||||
| @ -1 +0,0 @@ | |||||||
| {"version":3,"file":"test.js","sourceRoot":"","sources":["../src/test.ts"],"names":[],"mappings":";;AAAA,mCAA+C;AAC/C,mCAAqC;AAErC,eAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAA;AACnB,eAAO,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,EAAE,EAAE,CAAC,EAAE,IAAI,EAAE,EAAE,IAAI,CAAC,CAAC;AAC1C,eAAO,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,YAAY,CAAC,CAAC,CAAC;AACvC,eAAO,CAAC,KAAK,CAAC,kBAAkB,CAAC,CAAC;AAClC,eAAO,CAAC,YAAY,CAAC,GAAG,EAAE,IAAI,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;AAE/C,eAAO,CAAC,GAAG,CAAC,gFAAgF,CAAC,CAAA;AAE7F,IAAI,GAAG,GAAG,IAAI,KAAK,EAAE,CAAA;AACrB,IAAI,OAAO,GAAG,CAAC,KAAK,KAAK,QAAQ;IAAE,OAAO,CAAC,GAAG,CAAC,oBAAoB,EAAE,GAAG,CAAC,KAAK,CAAC,CAAA;AAE/E,IAAI,GAAG,GAAG,IAAI,mBAAW,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,CAAC,CAAC;AAC5C,GAAG,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAA;AAEnC,IAAI,IAAI,GAAG,IAAI,mBAAW,CAAC,OAAO,CAAC,CAAC;AACpC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AAEtC,IAAI,KAAK,GAAG,IAAI,mBAAW,CAAC,OAAO,CAAC,CAAC;AACrC,KAAK,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAA;AACxC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AACtC,KAAK,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAA;AACxC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AACtC,KAAK,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAA;AACxC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AACtC,KAAK,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAA;AACxC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AACtC,KAAK,CAAC,GAAG,CAAC,6BAA6B,CAAC,CAAA;AACxC,IAAI,CAAC,GAAG,CAAC,4BAA4B,CAAC,CAAA;AAEtC,eAAO,CAAC,WAAW,GAAG,KAAK,CAAC;AAC5B,KAAK,oBAAoB,KAAa,EAAE,YAAoB;IACzD,MAAM,eAAO,CAAC,YAAY,EAAE,CAAC;IAC7B,MAAM,QAAQ,GAAG,oBAAW,CAAC,YAAY,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAA;IAC1D,MAAM,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAC3B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,EAAE,CAAC,EAAE,EAAE;QAC7B,eAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAA;KACvB;IACD,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAC/B,MAAM,UAAU,GAAG,GAAG,CAAC;IACvB,MAAM,eAAO,CAAC,YAAY,EAAE,CAAC;IAC7B,MAAM,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC,GAAG,UAAU,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC;IAC1C,OAAO,CAAC,GAAG,CAAC,kBAAkB,EAAE,GAAG,OAAO,UAAU,KAAK,4BAA4B,YAAY,aAAa,CAAC,CAAC;IAChH,OAAO,CAAC,GAAG,CAAC,oBAAoB,CAAC,EAAE,GAAG,OAAO,CAAC,GAAG,KAAK,iBAAiB,CAAC,CAAA;AAC3E,CAAC;AAED,eAAO,CAAC,YAAY,EAAE,CAAC,IAAI,CAAC,KAAK,IAAI,EAAE;IACpC,OAAO,CAAC,GAAG,CAAC,uBAAuB,CAAC,CAAA;IACpC,MAAM,SAAS,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IAE7B,OAAO,CAAC,GAAG,CAAC,0BAA0B,CAAC,CAAA;IACvC,MAAM,SAAS,CAAC,MAAM,EAAE,GAAG,CAAC,CAAA;AAC/B,CAAC,CAAC,CAAC"} |  | ||||||
							
								
								
									
										1290
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
							
						
						
									
										1290
									
								
								package-lock.json
									
									
									
										generated
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										35
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										35
									
								
								package.json
									
									
									
									
									
								
							| @ -1,25 +1,42 @@ | |||||||
| { | { | ||||||
|    "name": "@hibas123/nodelogging", |    "name": "@hibas123/nodelogging", | ||||||
|    "version": "1.3.20", |    "packageManager": "yarn@3.6.4", | ||||||
|  |    "version": "4.0.1", | ||||||
|    "description": "", |    "description": "", | ||||||
|    "main": "out/index.js", |    "main": "out/index.js", | ||||||
|    "types": "out/index.d.ts", |    "types": "out/index.d.ts", | ||||||
|  |    "module": "esm/index.js", | ||||||
|    "scripts": { |    "scripts": { | ||||||
|       "prepublish": "tsc", |       "prepublish": "npm run build", | ||||||
|       "build": "tsc", |       "build": "tsc && tsc -p tsconfig.esm.json", | ||||||
|       "watch": "tsc --watch", |       "watch-ts": "tsc --watch", | ||||||
|       "test": "node out/test.js", |       "watch-js": "nodemon out/test.js", | ||||||
|  |       "watch": "concurrently npm:watch-*", | ||||||
|  |       "test": "npm run build && node out/test.js", | ||||||
|  |       "benchmark": "npm run build && node out/benchmark.js", | ||||||
|       "live": "nodemon out/test.js" |       "live": "nodemon out/test.js" | ||||||
|    }, |    }, | ||||||
|    "repository": { |    "repository": { | ||||||
|       "type": "git", |       "type": "git", | ||||||
|       "url": "https://git.stamm.me/PerfCloud/nodelogging.git" |       "url": "https://git.stamm.me/OpenServer/NodeLogging.git" | ||||||
|    }, |    }, | ||||||
|    "author": "Fabian Stamm", |    "author": "Fabian Stamm", | ||||||
|    "license": "MIT", |    "license": "MIT", | ||||||
|  |    "files": [ | ||||||
|  |       "src/", | ||||||
|  |       "out/", | ||||||
|  |       "esm/", | ||||||
|  |       "tsconfig.json", | ||||||
|  |       "readme.md" | ||||||
|  |    ], | ||||||
|    "devDependencies": { |    "devDependencies": { | ||||||
|       "@types/node": "^8.0.24", |       "@types/node": "^24.9.1", | ||||||
|       "nodemon": "^1.17.4", |       "concurrently": "^9.2.1", | ||||||
|       "typescript": "^2.4.2" |       "nodemon": "^3.1.10", | ||||||
|  |       "typescript": "^5.9.3" | ||||||
|  |    }, | ||||||
|  |    "dependencies": { | ||||||
|  |       "@hibas123/logging": "^4.0.2", | ||||||
|  |       "@hibas123/utils": "^2.2.18" | ||||||
|    } |    } | ||||||
| } | } | ||||||
							
								
								
									
										66
									
								
								readme.md
									
									
									
									
									
								
							
							
						
						
									
										66
									
								
								readme.md
									
									
									
									
									
								
							| @ -1,4 +1,8 @@ | |||||||
| Simple node logging module, that supports terminal coloring and writing to files | Simple logging module, that supports terminal coloring and writing to files. | ||||||
|  |  | ||||||
|  | This module build ontop of [@hibas123/logging](https://www.npmjs.com/package/@hibas123/utils). | ||||||
|  |  | ||||||
|  | It extends the default behavior to support Logging to Files out of the Box. | ||||||
|  |  | ||||||
| # Getting Started | # Getting Started | ||||||
|  |  | ||||||
| @ -27,23 +31,69 @@ All Logging types except the simple error take as many arguments as you want. Th | |||||||
|  |  | ||||||
| NodeLogging can work without any configuration, but it may be useful to change the log output folder. | NodeLogging can work without any configuration, but it may be useful to change the log output folder. | ||||||
|  |  | ||||||
| Todo so you are capable of creating own instances of the LoggingBase class | To do so you are capable of creating own instances of the LoggingBase class | ||||||
|  |  | ||||||
| ``` javascript | ``` javascript | ||||||
|    const CustomLogging = new LoggingBase({ |    const CustomLogging = new LoggingBase(name | { | ||||||
|       name: "custom", |       name: "custom", // default undefined | ||||||
|  |       files: true | false | { //default true | ||||||
|          logfile: "./logs/test.log", |          logfile: "./logs/test.log", | ||||||
|          errorfile: "/var/log/custom.err", |          errorfile: "/var/log/custom.err", | ||||||
|       console_out: false |       } | ||||||
|  |       console: false // default true | ||||||
|    }); |    }); | ||||||
| ``` | ``` | ||||||
|  |  | ||||||
| The name property prefixes the console output with the name. Also if no logfile or errorfile is created the following standard values are used: | The name property prefixes the console output with the name. Also if no logfile or errorfile is created the following standard values are used: | ||||||
|  |  | ||||||
| ./logs/all.{name}.log | - ./logs/all.{name}.log | ||||||
| ./logs/error.{name}.log | - ./logs/error.{name}.log | ||||||
|  |  | ||||||
|  | To not use any logfiles just set files to false. | ||||||
|  |  | ||||||
|  | # Plugins | ||||||
|  |  | ||||||
|  | There is a Plugin API available, that makes is possible to add custom Logging Adapter. | ||||||
|  |  | ||||||
|  | ``` javascript | ||||||
|  | const Demo = new LoggingExtended("Demo"); | ||||||
|  | Demo.addAdapter(new DemoAdapter({ color: "rainbow" })); | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | The adapters need to provide a very simple Interface: | ||||||
|  |  | ||||||
|  | ``` typescript | ||||||
|  | interface Adapter { | ||||||
|  |     init(observable: ObservableInterface<Message>, name?: string): void | Promise<void>; | ||||||
|  |  | ||||||
|  |     flush(sync: true): void; | ||||||
|  |     flush(sync: false): void | Promise<void>; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface Message { | ||||||
|  |     type: LoggingTypes; | ||||||
|  |     name?:string; | ||||||
|  |     text: { | ||||||
|  |     	raw: string[], | ||||||
|  |     	formatted: string[] | ||||||
|  |     }; | ||||||
|  |     date: Date; | ||||||
|  |     file: string; | ||||||
|  |     customColors?:string; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | enum LoggingTypes { | ||||||
|  |    Log, | ||||||
|  |    Warning, | ||||||
|  |    Error, | ||||||
|  |    Debug | ||||||
|  | } | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | The `ObservableInterface` comes from `@hibas123/utils`. It provides a very simple api for subscribing and unsubscribing from the message events. | ||||||
|  |  | ||||||
|  | More Details on Observable [here](https://git.stamm.me/OpenServer/Utils) | ||||||
|  |  | ||||||
| To not use any logfiles just set the values to null. |  | ||||||
| # License  | # License  | ||||||
| MIT | MIT | ||||||
|  |  | ||||||
|  | |||||||
							
								
								
									
										51
									
								
								src/benchmark.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										51
									
								
								src/benchmark.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,51 @@ | |||||||
|  | import { Formatted, LoggingBase, LoggingTypes } from "@hibas123/logging"; | ||||||
|  | import { once } from "events"; | ||||||
|  | import { createWriteStream } from "fs"; | ||||||
|  | import { FileAdapter } from "./filewriter"; | ||||||
|  |  | ||||||
|  | let results = {}; | ||||||
|  |  | ||||||
|  | async function benchmark( | ||||||
|  |    name: string, | ||||||
|  |    count: number, | ||||||
|  |    runner: (cnt: number) => Promise<void> | ||||||
|  | ) { | ||||||
|  |    console.log("Benchmark starting:", name); | ||||||
|  |    const start = process.hrtime.bigint(); | ||||||
|  |  | ||||||
|  |    await runner(count); | ||||||
|  |  | ||||||
|  |    const diffNS = process.hrtime.bigint() - start; | ||||||
|  |    const diffMS = Number(diffNS / BigInt(1000 * 1000)); | ||||||
|  |  | ||||||
|  |    console.log("Benchmark ended:", name); | ||||||
|  |  | ||||||
|  |    results[name] = { | ||||||
|  |       count, | ||||||
|  |       time: diffMS, | ||||||
|  |       timePerI: (diffMS / count).toFixed(4), | ||||||
|  |    }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | Promise.resolve().then(async () => { | ||||||
|  |    const largeText = "hallowelt!".repeat(250); | ||||||
|  |  | ||||||
|  |    await benchmark("large data", 100000, async (cnt) => { | ||||||
|  |       const lg = new LoggingBase({ | ||||||
|  |          console: false, | ||||||
|  |       }); | ||||||
|  |       const fs = new FileAdapter("logs/benchmark", Number.MAX_SAFE_INTEGER); | ||||||
|  |       await lg.addAdapter(fs); | ||||||
|  |  | ||||||
|  |       console.time("Logging"); | ||||||
|  |       for (let i = 0; i < cnt; i++) { | ||||||
|  |          lg.log(largeText); | ||||||
|  |       } | ||||||
|  |       console.timeEnd("Logging"); | ||||||
|  |  | ||||||
|  |       await fs.close(); | ||||||
|  |       await lg.close(); | ||||||
|  |    }); | ||||||
|  |  | ||||||
|  |    console.table(results); | ||||||
|  | }); | ||||||
							
								
								
									
										293
									
								
								src/filewriter.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										293
									
								
								src/filewriter.ts
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,293 @@ | |||||||
|  | import { AwaitStore, Lock } from "@hibas123/utils"; | ||||||
|  | import * as fs from "fs"; | ||||||
|  | import * as path from "path"; | ||||||
|  | import { Adapter, Message, Formatted, LoggingTypes } from "@hibas123/logging"; | ||||||
|  | import { once } from "events"; | ||||||
|  |  | ||||||
|  | const MAX_FILE_SIZE = 500000000; | ||||||
|  |  | ||||||
|  | export class FileAdapter implements Adapter { | ||||||
|  |    level = LoggingTypes.Debug; | ||||||
|  |  | ||||||
|  |    file: Files; | ||||||
|  |    isInit = new AwaitStore(false); | ||||||
|  |    constructor(private filename: string, private maxFileSize = MAX_FILE_SIZE) {} | ||||||
|  |  | ||||||
|  |    setLevel(level: LoggingTypes) { | ||||||
|  |       this.level = level; | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    async init() { | ||||||
|  |       if (!this.file) { | ||||||
|  |          this.file = Files.getFile(this.filename); | ||||||
|  |          await this.file | ||||||
|  |             .init(this.maxFileSize) | ||||||
|  |             .then(() => this.isInit.send(true)); | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    flush(sync: boolean) { | ||||||
|  |       // return this.file.flush(sync); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    onMessage(message: Message) { | ||||||
|  |       let msg = Buffer.from(Formatted.strip(message.text) + "\n"); | ||||||
|  |       this.file.write(msg); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    async close() { | ||||||
|  |       if (this.file) { | ||||||
|  |          await this.file.close(); | ||||||
|  |          this.file = undefined; | ||||||
|  |       } | ||||||
|  |       this.isInit.send(false); | ||||||
|  |    } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | //TODO: Optimise write path | ||||||
|  |  | ||||||
|  | const Debounce = (callback: () => void, iv = 500, max = 100) => { | ||||||
|  |    let to: any; | ||||||
|  |    let curr = 0; | ||||||
|  |  | ||||||
|  |    return { | ||||||
|  |       trigger: () => { | ||||||
|  |          curr++; | ||||||
|  |          if (curr >= max) { | ||||||
|  |             curr = 0; // not clearing timeout, since this is a very high cost operation | ||||||
|  |             callback(); | ||||||
|  |          } else if (!to) { | ||||||
|  |             to = setTimeout(() => { | ||||||
|  |                to = undefined; | ||||||
|  |                curr = 0; | ||||||
|  |                callback(); | ||||||
|  |             }, iv); | ||||||
|  |          } | ||||||
|  |       }, | ||||||
|  |    }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const QUEUE_START_SIZE = 10000; | ||||||
|  |  | ||||||
|  | export class Files { | ||||||
|  |    private static files = new Map<string, Files>(); | ||||||
|  |    static getFile(filename: string): Files { | ||||||
|  |       filename = path.resolve(filename); | ||||||
|  |       let file = this.files.get(filename); | ||||||
|  |       if (!file) { | ||||||
|  |          file = new Files(filename); | ||||||
|  |          this.files.set(filename, file); | ||||||
|  |       } | ||||||
|  |       file.open++; | ||||||
|  |       return file; | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private open = 0; | ||||||
|  |  | ||||||
|  |    #maxFileSize = MAX_FILE_SIZE; | ||||||
|  |    #size: number = 0; | ||||||
|  |    #stream: fs.WriteStream = undefined; | ||||||
|  |    #lock = new Lock(); | ||||||
|  |    #debounce = Debounce(this.checkQueue.bind(this)); | ||||||
|  |    #initialized = false; | ||||||
|  |    #queue: Buffer[] = new Array(QUEUE_START_SIZE); | ||||||
|  |    #queueIdx = 0; | ||||||
|  |  | ||||||
|  |    public get initlialized() { | ||||||
|  |       return this.#initialized; | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private constructor(private file: string) {} | ||||||
|  |  | ||||||
|  |    public async init(maxFileSize: number) { | ||||||
|  |       if (this.#initialized) return; | ||||||
|  |       this.#maxFileSize = maxFileSize; | ||||||
|  |  | ||||||
|  |       let lock = await this.#lock.getLock(); | ||||||
|  |  | ||||||
|  |       const folder = path.dirname(this.file); | ||||||
|  |       if (folder) { | ||||||
|  |          if (!(await fsExists(folder))) { | ||||||
|  |             await fsMkDir(folder).catch(() => {}); //Could happen, if two seperate instances want to create the same folder so ignoring | ||||||
|  |          } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       await this.initializeFile(); | ||||||
|  |       this.#initialized = true; | ||||||
|  |       await this.checkQueue(true); | ||||||
|  |       lock.release(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private async initializeFile(new_file = false) { | ||||||
|  |       try { | ||||||
|  |          if (this.#stream) { | ||||||
|  |             const closePrms = once(this.#stream, "close"); | ||||||
|  |  | ||||||
|  |             this.#stream.end(); | ||||||
|  |  | ||||||
|  |             await closePrms; | ||||||
|  |          } | ||||||
|  |  | ||||||
|  |          let size = 0; | ||||||
|  |          if (await fsExists(this.file)) { | ||||||
|  |             let stats = await fsStat(this.file); | ||||||
|  |             if (new_file || stats.size >= this.#maxFileSize) { | ||||||
|  |                if (await fsExists(this.file + ".old")) | ||||||
|  |                   await fsUnlink(this.file + ".old"); | ||||||
|  |                await fsMove(this.file, this.file + ".old"); | ||||||
|  |             } else { | ||||||
|  |                size = stats.size; | ||||||
|  |             } | ||||||
|  |          } | ||||||
|  |  | ||||||
|  |          this.#stream = fs.createWriteStream(this.file, { flags: "a" }); | ||||||
|  |          this.#size = size; | ||||||
|  |       } catch (err) { | ||||||
|  |          console.log(err); | ||||||
|  |          //TODO: is this the right behavior? Probably not... | ||||||
|  |          process.exit(1); | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private async checkQueue(nolock: boolean = false) { | ||||||
|  |       let lock: any; | ||||||
|  |       if (nolock == false) { | ||||||
|  |          //TODO: New design might cause new messages to be "stalled" till close or another message | ||||||
|  |          if (this.#lock.locked) return; | ||||||
|  |          lock = await this.#lock.getLock(); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       const queue = this.#queue; | ||||||
|  |       const queueCnt = this.#queueIdx; | ||||||
|  |  | ||||||
|  |       this.#queue = new Array(QUEUE_START_SIZE); | ||||||
|  |       this.#queueIdx = 0; | ||||||
|  |  | ||||||
|  |       let buffer = Buffer.alloc(1024 * 128); | ||||||
|  |       let ci = 0; | ||||||
|  |       for (let i = 0; i < queueCnt; i++) { | ||||||
|  |          const entry = queue[i]; | ||||||
|  |  | ||||||
|  |          if (entry.length + ci > buffer.length) { | ||||||
|  |             await this.write_to_file(buffer.slice(0, ci)); | ||||||
|  |             ci = 0; | ||||||
|  |             if (entry.length > buffer.length) { | ||||||
|  |                await this.write_to_file(entry); | ||||||
|  |             } else { | ||||||
|  |                entry.copy(buffer, ci); | ||||||
|  |                ci += entry.length; | ||||||
|  |             } | ||||||
|  |          } else { | ||||||
|  |             entry.copy(buffer, ci); | ||||||
|  |             ci += entry.length; | ||||||
|  |          } | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (ci > 0) { | ||||||
|  |          await this.write_to_file(buffer.slice(0, ci)); | ||||||
|  |       } | ||||||
|  |  | ||||||
|  |       if (lock) lock.release(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    public async close() { | ||||||
|  |       //TODO: maybe some raceconditions when open collides with close | ||||||
|  |       const lock = await this.#lock.getLock(); | ||||||
|  |       await this.checkQueue(true); | ||||||
|  |       this.open--; | ||||||
|  |       if (this.open <= 0) { | ||||||
|  |          const a = once(this.#stream, "close"); | ||||||
|  |          this.#stream.close(); | ||||||
|  |          await a; | ||||||
|  |          Files.files.delete(this.file); | ||||||
|  |       } | ||||||
|  |       lock.release(); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    private async write_to_file(data: Buffer) { | ||||||
|  |       try { | ||||||
|  |          if ( | ||||||
|  |             data.byteLength < this.#maxFileSize && | ||||||
|  |             this.#size + data.byteLength > this.#maxFileSize | ||||||
|  |          ) { | ||||||
|  |             await this.initializeFile(true); | ||||||
|  |          } | ||||||
|  |          this.#size += data.byteLength; | ||||||
|  |          this.#stream.write(data); | ||||||
|  |       } catch (err) { | ||||||
|  |          // TODO: Better error handling! | ||||||
|  |          console.error(err); | ||||||
|  |          this.initializeFile(false); | ||||||
|  |          this.write_to_file(data); | ||||||
|  |       } | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    public write(data: Buffer) { | ||||||
|  |       this.#queue[this.#queueIdx++] = data; | ||||||
|  |       this.#debounce.trigger(); | ||||||
|  |    } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function fsUnlink(path: string) { | ||||||
|  |    if (fs.promises?.unlink) { | ||||||
|  |       return fs.promises.unlink(path); | ||||||
|  |    } | ||||||
|  |    return new Promise<void>((resolve, reject) => { | ||||||
|  |       fs.unlink(path, (err) => { | ||||||
|  |          if (err) reject(err); | ||||||
|  |          else resolve(); | ||||||
|  |       }); | ||||||
|  |    }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function fsStat(path: string) { | ||||||
|  |    if (fs.promises?.stat) { | ||||||
|  |       return fs.promises.stat(path); | ||||||
|  |    } | ||||||
|  |    return new Promise<fs.Stats>((resolve, reject) => { | ||||||
|  |       fs.stat(path, (err, stats) => { | ||||||
|  |          if (err) reject(err); | ||||||
|  |          else resolve(stats); | ||||||
|  |       }); | ||||||
|  |    }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function fsMove(oldPath: string, newPath: string) { | ||||||
|  |    return new Promise<void>((resolve, reject) => { | ||||||
|  |       let callback = (err?) => { | ||||||
|  |          if (err) reject(err); | ||||||
|  |          else resolve(); | ||||||
|  |       }; | ||||||
|  |  | ||||||
|  |       fs.rename(oldPath, newPath, function (err) { | ||||||
|  |          if (err) { | ||||||
|  |             if (err.code === "EXDEV") { | ||||||
|  |                copy(); | ||||||
|  |             } else { | ||||||
|  |                callback(err); | ||||||
|  |             } | ||||||
|  |             return; | ||||||
|  |          } | ||||||
|  |          callback(); | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       function copy() { | ||||||
|  |          fs.copyFile(oldPath, newPath, (err) => { | ||||||
|  |             if (err) callback(err); | ||||||
|  |             else fs.unlink(oldPath, callback); | ||||||
|  |          }); | ||||||
|  |       } | ||||||
|  |    }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function fsExists(path: string) { | ||||||
|  |    return new Promise<boolean>((resolve, reject) => { | ||||||
|  |       fs.access(path, (err) => resolve(!err)); | ||||||
|  |    }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function fsMkDir(path: string) { | ||||||
|  |    return new Promise<void>((resolve, reject) => { | ||||||
|  |       fs.mkdir(path, (err) => (err ? reject(err) : resolve())); | ||||||
|  |    }); | ||||||
|  | } | ||||||
							
								
								
									
										465
									
								
								src/index.ts
									
									
									
									
									
								
							
							
						
						
									
										465
									
								
								src/index.ts
									
									
									
									
									
								
							| @ -1,451 +1,36 @@ | |||||||
| import * as util from "util"; | export { FileAdapter } from "./filewriter"; | ||||||
| import * as fs from "fs"; | import { FileAdapter } from "./filewriter"; | ||||||
| import { EventEmitter } from "events"; | import { LoggingBase } from "@hibas123/logging"; | ||||||
| import * as path from "path"; | import Logging from "@hibas123/logging"; | ||||||
| import Lock from "./lock"; |  | ||||||
|  |  | ||||||
| export const Colors = { | LoggingBase.nativeFunctions = { | ||||||
|    Reset: "\x1b[0m", |    startTimer: () => { | ||||||
|    Bright: "\x1b[1m", |       if (process.hrtime.bigint) { | ||||||
|    Dim: "\x1b[2m", |          return process.hrtime.bigint(); | ||||||
|    Underscore: "\x1b[4m", |  | ||||||
|    Blink: "\x1b[5m", |  | ||||||
|    Reverse: "\x1b[7m", |  | ||||||
|    Hidden: "\x1b[8m", |  | ||||||
|  |  | ||||||
|    FgBlack: "\x1b[30m", |  | ||||||
|    FgRed: "\x1b[31m", |  | ||||||
|    FgGreen: "\x1b[32m", |  | ||||||
|    FgYellow: "\x1b[33m", |  | ||||||
|    FgBlue: "\x1b[34m", |  | ||||||
|    FgMagenta: "\x1b[35m", |  | ||||||
|    FgCyan: "\x1b[36m", |  | ||||||
|    FgWhite: "\x1b[37m", |  | ||||||
|  |  | ||||||
|    BgBlack: "\x1b[40m", |  | ||||||
|    BgRed: "\x1b[41m", |  | ||||||
|    BgGreen: "\x1b[42m", |  | ||||||
|    BgYellow: "\x1b[43m", |  | ||||||
|    BgBlue: "\x1b[44m", |  | ||||||
|    BgMagenta: "\x1b[45m", |  | ||||||
|    BgCyan: "\x1b[46m", |  | ||||||
|    BgWhite: "\x1b[47m" |  | ||||||
| } |  | ||||||
|  |  | ||||||
| const maxFileSize = 500000000; |  | ||||||
|  |  | ||||||
| const OriginalErrorStackFunction = (<any>Error.prototype).prepareStackTrace |  | ||||||
|  |  | ||||||
| export interface LoggingBaseOptions { |  | ||||||
|    /** |  | ||||||
|     * Name will be prefixed on Console output and added to logfiles, if not specified here |  | ||||||
|     */ |  | ||||||
|    name: string, |  | ||||||
|    /** |  | ||||||
|     * Filename/path of the logfile. Skip if generated with name |  | ||||||
|     */ |  | ||||||
|    logfile: string; |  | ||||||
|    /** |  | ||||||
|     * Filename/path of the logfile. Skip if generated with name |  | ||||||
|     */ |  | ||||||
|    errorfile: string; |  | ||||||
|    /** |  | ||||||
|     * Prints output to console |  | ||||||
|     */ |  | ||||||
|    console_out: boolean; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| class LoggingFiles { |  | ||||||
|    private static files: LoggingFiles[] = []; |  | ||||||
|    static getFile(filename: string): LoggingFiles { |  | ||||||
|       filename = path.resolve(filename); |  | ||||||
|       let file = this.files.find(e => e.file === filename); |  | ||||||
|       if (!file) { |  | ||||||
|          file = new LoggingFiles(filename); |  | ||||||
|          this.files.push(file); |  | ||||||
|       } |  | ||||||
|       return file; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private file: string; |  | ||||||
|    private size: number = 0; |  | ||||||
|    private stream: fs.WriteStream = undefined; |  | ||||||
|    private lock = new Lock(); |  | ||||||
|    private constructor(file: string) { |  | ||||||
|       this.file = path.resolve(file); |  | ||||||
|       this.init(); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    public async awaitinit() { |  | ||||||
|       (await this.lock.getLock()).release(); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private async init() { |  | ||||||
|       let lock = await this.lock.getLock(); |  | ||||||
|       await this.initializeFile() |  | ||||||
|       lock.release(); |  | ||||||
|       this.checkQueue() |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private async initializeFile(new_file = false) { |  | ||||||
|       try { |  | ||||||
|          if (this.stream) { |  | ||||||
|             this.stream.close(); |  | ||||||
|          } |  | ||||||
|          const folder = path.dirname(this.file); |  | ||||||
|          if (folder) { |  | ||||||
|             if (!await fsExists(folder)) { |  | ||||||
|                await fsMkDir(folder).catch(() => { }); //Could happen, if two seperate instances want to create the same folder so ignoring |  | ||||||
|             } |  | ||||||
|          } |  | ||||||
|  |  | ||||||
|          let size = 0; |  | ||||||
|          if (await fsExists(this.file)) { |  | ||||||
|             let stats = await fsStat(this.file); |  | ||||||
|             if (new_file || stats.size >= maxFileSize) { |  | ||||||
|                if (await fsExists(this.file + ".old")) |  | ||||||
|                   await fsUnlink(this.file + ".old"); |  | ||||||
|                await fsMove(this.file, this.file + ".old") |  | ||||||
|       } else { |       } else { | ||||||
|                size = stats.size; |          return process.hrtime(); | ||||||
|       } |       } | ||||||
|          } |    }, | ||||||
|  |    diffTime: (start, end) => { | ||||||
|          this.stream = fs.createWriteStream(this.file, { flags: "a" }) |       if (process.hrtime.bigint) { | ||||||
|          this.size = size; |          return Number((BigInt(end) - BigInt(start)) / BigInt(1000)) / 1000; | ||||||
|       } catch (e) { |  | ||||||
|          console.log(e); |  | ||||||
|          //ToDo is this the right behavior? |  | ||||||
|          process.exit(1); |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private queue: Buffer[] = []; |  | ||||||
|  |  | ||||||
|    async checkQueue() { |  | ||||||
|       if (this.lock.locked) return; |  | ||||||
|       let lock = await this.lock.getLock(); |  | ||||||
|       let msg: Buffer; |  | ||||||
|       while (msg = this.queue.shift()) { |  | ||||||
|          await this.write_to_file(msg); |  | ||||||
|       } |  | ||||||
|       lock.release(); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private async write_to_file(data: Buffer) { |  | ||||||
|       try { |  | ||||||
|          if (data.byteLength < maxFileSize && this.size + data.byteLength > maxFileSize) { |  | ||||||
|             await this.initializeFile(true) |  | ||||||
|          } |  | ||||||
|          this.size += data.byteLength; |  | ||||||
|          this.stream.write(data); |  | ||||||
|       } catch (err) { |  | ||||||
|          console.error(err); |  | ||||||
|          this.initializeFile(false); |  | ||||||
|          this.write_to_file(data); |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    public write(data: Buffer) { |  | ||||||
|       this.queue.push(data); |  | ||||||
|       this.checkQueue() |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export class LoggingBase { |  | ||||||
|    private config: LoggingBaseOptions; |  | ||||||
|  |  | ||||||
|    private logFile: LoggingFiles; |  | ||||||
|    private errorFile: LoggingFiles; |  | ||||||
|  |  | ||||||
|    constructor(options?: Partial<LoggingBaseOptions> | string) { |  | ||||||
|       let opt: Partial<LoggingBaseOptions>; |  | ||||||
|       if (!options) opt = {} |  | ||||||
|       else if (typeof options === "string") { |  | ||||||
|          opt = { name: options }; |  | ||||||
|       } else { |       } else { | ||||||
|          opt = options; |          let diff = process.hrtime(start); | ||||||
|  |          return diff[0] * 1000 + diff[1] / 1000000; | ||||||
|       } |       } | ||||||
|       if (opt.name) { |    }, | ||||||
|          if (opt.logfile === undefined) { |    endTimer: (start) => { | ||||||
|             opt.logfile = `./logs/all.${opt.name}.log` |       if (process.hrtime.bigint) { | ||||||
|          } |          return Number((process.hrtime.bigint() - start) / BigInt(1000)) / 1000; | ||||||
|  |  | ||||||
|          if (opt.errorfile === undefined) { |  | ||||||
|             opt.errorfile = `./logs/error.${opt.name}.log` |  | ||||||
|          } |  | ||||||
|       } |  | ||||||
|       this.config = Object.assign(<LoggingBaseOptions>{ |  | ||||||
|          name: undefined, |  | ||||||
|          console_out: true, |  | ||||||
|          logfile: "./logs/all.log", |  | ||||||
|          errorfile: "./logs/error.log" |  | ||||||
|       }, opt); |  | ||||||
|  |  | ||||||
|       for (let key in this) { |  | ||||||
|          if (typeof this[key] === "function") this[key] = (<any>this[key]).bind(this); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (this.config.logfile) { |  | ||||||
|          this.logFile = LoggingFiles.getFile(this.config.logfile); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       if (this.config.errorfile) { |  | ||||||
|          this.errorFile = LoggingFiles.getFile(this.config.errorfile); |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    get console_out() { |  | ||||||
|       return this.config.console_out; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    set console_out(value: boolean) { |  | ||||||
|       this.config.console_out = value; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    public waitForSetup() { |  | ||||||
|       let w = []; |  | ||||||
|       if (this.logFile) w.push(this.logFile.awaitinit()); |  | ||||||
|       if (this.errorFile) w.push(this.errorFile.awaitinit()); |  | ||||||
|       return Promise.all(w) |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    public events: EventEmitter = new EventEmitter(); |  | ||||||
|  |  | ||||||
|    debug(...message: any[]) { |  | ||||||
|       this.message(LoggingTypes.Debug, message); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    log(...message: any[]) { |  | ||||||
|       this.message(LoggingTypes.Log, message); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    warning(...message: any[]) { |  | ||||||
|       this.message(LoggingTypes.Warning, message); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    logWithCustomColors(type: LoggingTypes, colors: string, ...message: any[]) { |  | ||||||
|       this.message(type, message, colors); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    error(error: Error | string) { |  | ||||||
|       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]); |  | ||||||
|       } else { |       } else { | ||||||
|          this.message(LoggingTypes.Error, [error.message, "\n", error.stack], undefined, getCallerFromExisting(error)); |          let diff = process.hrtime(start); | ||||||
|       } |          return diff[0] * 1000 + diff[1] / 1000000; | ||||||
|       } |       } | ||||||
|  |    }, | ||||||
|  | }; | ||||||
|  |  | ||||||
|    errorMessage(...message: any[]) { | export const DefaultFileAdapter = new FileAdapter("./logs/all.log"); | ||||||
|       this.message(LoggingTypes.Error, message); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private message(type: LoggingTypes, message: any[] | string, customColors?: string, caller?: { file: string, line: number }) { | Logging.addAdapter(DefaultFileAdapter); | ||||||
|       var consoleLogFormat = Colors.Reset; |  | ||||||
|       if (!customColors) { |  | ||||||
|          switch (type) { |  | ||||||
|             case LoggingTypes.Log: |  | ||||||
|                //m += FgWhite + BgBlack; |  | ||||||
|                break; |  | ||||||
|             case LoggingTypes.Error: |  | ||||||
|                consoleLogFormat += Colors.FgRed;//FgWhite + BgRed + FgWhite; |  | ||||||
|                break; |  | ||||||
|             case LoggingTypes.Debug: |  | ||||||
|                consoleLogFormat += Colors.FgCyan; |  | ||||||
|                break; |  | ||||||
|             case LoggingTypes.Warning: |  | ||||||
|                consoleLogFormat += Colors.FgYellow; |  | ||||||
|                break; |  | ||||||
|          } |  | ||||||
|       } else { |  | ||||||
|          consoleLogFormat += customColors; |  | ||||||
|       } |  | ||||||
|       var mb = ""; |  | ||||||
|       if (typeof message === "string") { |  | ||||||
|          mb = message; |  | ||||||
|       } else { |  | ||||||
|          message.forEach((e, i) => { |  | ||||||
|             if (typeof e !== "string") e = util.inspect(e, false, null); |  | ||||||
|             if (e.endsWith("\n") || i === message.length - 1) { |  | ||||||
|                mb += e; |  | ||||||
|             } else { |  | ||||||
|                mb += e + " "; |  | ||||||
|             } |  | ||||||
|          }); |  | ||||||
|       } |  | ||||||
|       let file = caller || getCallerFile(); |  | ||||||
|       let date = new Date().toISOString().replace(/T/, ' ').replace(/\..+/, ''); |  | ||||||
|       let prefix = `[${LoggingTypes[type]}][${file.file}:${file.line}][${date}]: `; |  | ||||||
|       let message_lines = mb.split("\n").map(line => prefix + line); |  | ||||||
|  |  | ||||||
|       if (this.config.console_out) { |  | ||||||
|          let name = ""; |  | ||||||
|          if (this.config.name) name = `[${this.config.name}]`; |  | ||||||
|          message_lines.forEach(line => console.log(consoleLogFormat + name + line + Colors.Reset)); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       let m = message_lines.join("\n"); |  | ||||||
|  |  | ||||||
|       m = m.replace(/[\u001b\u009b][[()#;?]*(?:[0-9]{1,4}(?:;[0-9]{0,4})*)?[0-9A-ORZcf-nqry=><]/g, ""); |  | ||||||
|  |  | ||||||
|       let index = m.indexOf("\x1b"); |  | ||||||
|       while (index >= 0) { |  | ||||||
|          m = m.substring(0, index) + m.substring(index + 5, m.length); |  | ||||||
|          index = m.indexOf("\x1b"); |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       let data = Buffer.from(m + "\n") |  | ||||||
|       if (type === LoggingTypes.Error && this.errorFile) { |  | ||||||
|          this.errorFile.write(data); |  | ||||||
|       } |  | ||||||
|       if (this.logFile) { |  | ||||||
|          this.logFile.write(data); |  | ||||||
|       } |  | ||||||
|       this.events.emit("message", { type: type, message: m, module: this.config.name }); |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export let Logging: LoggingBase = undefined; |  | ||||||
| if (process.env.LOGGING_NO_DEFAULT !== "true") { |  | ||||||
|    Logging = new LoggingBase(); |  | ||||||
| } |  | ||||||
| export default Logging; | export default Logging; | ||||||
|  |  | ||||||
| function fsUnlink(path) { |  | ||||||
|    return new Promise((resolve, reject) => { |  | ||||||
|       fs.unlink(path, (err) => { |  | ||||||
|          if (err) reject(err); |  | ||||||
|          else resolve(); |  | ||||||
|       }) |  | ||||||
|    }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function fsStat(path: string) { |  | ||||||
|    return new Promise<fs.Stats>((resolve, reject) => { |  | ||||||
|       fs.stat(path, (err, stats) => { |  | ||||||
|          if (err) reject(err); |  | ||||||
|          else resolve(stats); |  | ||||||
|       }) |  | ||||||
|    }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function fsMove(oldPath: string, newPath: string) { |  | ||||||
|    return new Promise((resolve, reject) => { |  | ||||||
|       let callback = (err?) => { |  | ||||||
|          if (err) reject(err) |  | ||||||
|          else resolve() |  | ||||||
|       } |  | ||||||
|  |  | ||||||
|       fs.rename(oldPath, newPath, function (err) { |  | ||||||
|          if (err) { |  | ||||||
|             if (err.code === 'EXDEV') { |  | ||||||
|                copy(); |  | ||||||
|             } else { |  | ||||||
|                callback(err) |  | ||||||
|             } |  | ||||||
|             return; |  | ||||||
|          } |  | ||||||
|          callback() |  | ||||||
|       }); |  | ||||||
|  |  | ||||||
|       function copy() { |  | ||||||
|          fs.copyFile(oldPath, newPath, (err) => { |  | ||||||
|             if (err) callback(err) |  | ||||||
|             else fs.unlink(oldPath, callback); |  | ||||||
|          }) |  | ||||||
|          // var readStream = fs.createReadStream(oldPath); |  | ||||||
|          // var writeStream = fs.createWriteStream(newPath); |  | ||||||
|  |  | ||||||
|          // readStream.on('error', callback); |  | ||||||
|          // writeStream.on('error', callback); |  | ||||||
|  |  | ||||||
|          // readStream.on('close', function () { |  | ||||||
|          //    fs.unlink(oldPath, callback); |  | ||||||
|          // }); |  | ||||||
|  |  | ||||||
|          // readStream.pipe(writeStream); |  | ||||||
|       } |  | ||||||
|    }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function fsExists(path: string) { |  | ||||||
|    return new Promise<boolean>((resolve, reject) => { |  | ||||||
|       fs.exists(path, resolve); |  | ||||||
|    }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function fsMkDir(path: string) { |  | ||||||
|    return new Promise((resolve, reject) => { |  | ||||||
|       fs.mkdir(path, (err) => err ? reject(err) : resolve()); |  | ||||||
|    }); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getStack() { |  | ||||||
|    // Save original Error.prepareStackTrace |  | ||||||
|    let origPrepareStackTrace = (<any>Error).prepareStackTrace; |  | ||||||
|  |  | ||||||
|    // Override with function that just returns `stack` |  | ||||||
|    (<any>Error).prepareStackTrace = function (_, stack) { |  | ||||||
|       return stack |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    // Create a new `Error`, which automatically gets `stack` |  | ||||||
|    let err = new Error(); |  | ||||||
|  |  | ||||||
|    // Evaluate `err.stack`, which calls our new `Error.prepareStackTrace` |  | ||||||
|    let stack: any[] = <any>err.stack; |  | ||||||
|  |  | ||||||
|    // Restore original `Error.prepareStackTrace` |  | ||||||
|    (<any>Error).prepareStackTrace = origPrepareStackTrace; |  | ||||||
|  |  | ||||||
|    // Remove superfluous function call on stack |  | ||||||
|    stack.shift(); // getStack --> Error |  | ||||||
|  |  | ||||||
|    return stack |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCallerFile() { |  | ||||||
|    try { |  | ||||||
|       let stack = getStack() |  | ||||||
|  |  | ||||||
|       let current_file = stack.shift().getFileName(); |  | ||||||
|  |  | ||||||
|       while (stack.length) { |  | ||||||
|          let caller_file = stack.shift(); |  | ||||||
|          const util = require("util") |  | ||||||
|          if (current_file !== caller_file.getFileName()) |  | ||||||
|             return { |  | ||||||
|                file: path.basename(caller_file.getFileName()), |  | ||||||
|                line: caller_file.getLineNumber() |  | ||||||
|             }; |  | ||||||
|       } |  | ||||||
|    } catch (err) { } |  | ||||||
|    return { file: undefined, line: 0 }; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function getCallerFromExisting(err: Error): { file: string, line: number } { |  | ||||||
|    let lines = err.stack.split("\n"); |  | ||||||
|    let current = path.basename(__filename); |  | ||||||
|    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) |  | ||||||
|       if (matches && matches.length > 0) { |  | ||||||
|          let [f, line] = matches[0].split(":") |  | ||||||
|          if (f != current) { |  | ||||||
|             return { |  | ||||||
|                file: f, line: Number(line) |  | ||||||
|             }; |  | ||||||
|          } |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export enum LoggingTypes { |  | ||||||
|    Log, |  | ||||||
|    Warning, |  | ||||||
|    Error, |  | ||||||
|    Debug |  | ||||||
| } |  | ||||||
							
								
								
									
										36
									
								
								src/lock.ts
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								src/lock.ts
									
									
									
									
									
								
							| @ -1,36 +0,0 @@ | |||||||
| export type Release = { release: () => void }; |  | ||||||
| export default class Lock { |  | ||||||
|    private _locked: boolean = false; |  | ||||||
|    get locked() { |  | ||||||
|       return this._locked; |  | ||||||
|    } |  | ||||||
|    private toCome: (() => void)[] = []; |  | ||||||
|  |  | ||||||
|    constructor() { |  | ||||||
|       this.release = this.release.bind(this); |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    async getLock(): Promise<Release> { |  | ||||||
|       if (!this._locked) return { release: this.lock() }; |  | ||||||
|       else { |  | ||||||
|          return new Promise<Release>((resolve) => { |  | ||||||
|             this.toCome.push(() => { |  | ||||||
|                resolve({ release: this.lock() }); |  | ||||||
|             }) |  | ||||||
|          }) |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private lock() { |  | ||||||
|       this._locked = true; |  | ||||||
|       return this.release; |  | ||||||
|    } |  | ||||||
|  |  | ||||||
|    private async release() { |  | ||||||
|       if (this.toCome.length > 0) { |  | ||||||
|          this.toCome.shift()(); |  | ||||||
|       } else { |  | ||||||
|          this._locked = false; |  | ||||||
|       } |  | ||||||
|    } |  | ||||||
| } |  | ||||||
							
								
								
									
										112
									
								
								src/test.ts
									
									
									
									
									
								
							
							
						
						
									
										112
									
								
								src/test.ts
									
									
									
									
									
								
							| @ -1,55 +1,103 @@ | |||||||
| import { Logging, LoggingBase } from "./index"; |  | ||||||
| import { randomBytes } from "crypto"; | import { randomBytes } from "crypto"; | ||||||
|  | import * as fs from "fs"; | ||||||
|  | import { LoggingBase } from "@hibas123/logging"; | ||||||
|  | import Logging, { DefaultFileAdapter, FileAdapter } from "."; | ||||||
|  |  | ||||||
| Logging.log("test") | const deleteFolderRecursive = function (path: string) { | ||||||
|  |    if (fs.existsSync(path)) { | ||||||
|  |       fs.readdirSync(path).forEach(function (file, index) { | ||||||
|  |          var curPath = path + "/" + file; | ||||||
|  |          if (fs.lstatSync(curPath).isDirectory()) { | ||||||
|  |             // recurse | ||||||
|  |             deleteFolderRecursive(curPath); | ||||||
|  |          } else { | ||||||
|  |             // delete file | ||||||
|  |             fs.unlinkSync(curPath); | ||||||
|  |          } | ||||||
|  |       }); | ||||||
|  |       fs.rmdirSync(path); | ||||||
|  |    } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | deleteFolderRecursive("./logs"); | ||||||
|  |  | ||||||
|  | Logging.log("test"); | ||||||
| Logging.log("i", "am", { a: "an" }, 1000); | Logging.log("i", "am", { a: "an" }, 1000); | ||||||
| Logging.error(new Error("fehler 001")); | Logging.error(new Error("fehler 001")); | ||||||
| Logging.debug("Some Debug infos"); | Logging.debug("Some Debug infos"); | ||||||
| Logging.errorMessage("i", "am", "an", "error"); | Logging.error("i", "am", "an", "error"); | ||||||
|  |  | ||||||
| Logging.log("\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" | ||||||
|  | ); | ||||||
|  |  | ||||||
| let err = new Error() | let err = new Error(); | ||||||
| if (typeof err.stack !== "string") console.log("Stacktrace invalid", err.stack) | if (typeof err.stack !== "string") console.log("Stacktrace invalid", err.stack); | ||||||
|  |  | ||||||
| let cus = new LoggingBase({ name: "test" }); | let cus = new LoggingBase({ name: "test" }); | ||||||
| cus.log("Hello from custom Logger") | cus.log("Hello from custom Logger"); | ||||||
|  |  | ||||||
| let cus2 = new LoggingBase("test2"); | let cus2 = Logging.getChild("test2"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
|  |  | ||||||
| let cus22 = new LoggingBase("test2"); | let cus22 = Logging.getChild("test2"); | ||||||
| cus22.log("Hello from custom Logger 22") | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
| cus22.log("Hello from custom Logger 22") | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
| cus22.log("Hello from custom Logger 22") | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
| cus22.log("Hello from custom Logger 22") | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
| cus22.log("Hello from custom Logger 22") | cus22.log("Hello from custom Logger 22"); | ||||||
| cus2.log("Hello from custom Logger 2") | cus2.log("Hello from custom Logger 2"); | ||||||
|  |  | ||||||
| Logging.console_out = false; |  | ||||||
| async function benchmark(count: number, message_size: number) { | async function benchmark(count: number, message_size: number) { | ||||||
|    await Logging.waitForSetup(); |    const BenchmarkLogger = new LoggingBase({ | ||||||
|    const randData = randomBytes(message_size).toString("hex") |       console: false, | ||||||
|  |       name: "bench", | ||||||
|  |    }); | ||||||
|  |  | ||||||
|  |    if (fs.existsSync("logs/benchmark")) { | ||||||
|  |       fs.unlinkSync("logs/benchmark"); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    if (fs.existsSync("logs/benchmark.old")) { | ||||||
|  |       fs.unlinkSync("logs/benchmark.old"); | ||||||
|  |    } | ||||||
|  |  | ||||||
|  |    const BenchmarkFile = new FileAdapter("logs/benchmark"); | ||||||
|  |  | ||||||
|  |    BenchmarkLogger.addAdapter(BenchmarkFile); | ||||||
|  |  | ||||||
|  |    const randData = randomBytes(message_size / 2).toString("hex"); | ||||||
|    const t = process.hrtime(); |    const t = process.hrtime(); | ||||||
|    for (let i = 0; i < count; i++) { |    for (let i = 0; i < count; i++) { | ||||||
|       Logging.log(randData) |       BenchmarkLogger.log(randData); | ||||||
|    } |    } | ||||||
|  |    await BenchmarkFile.flush(false); | ||||||
|  |    await BenchmarkLogger.close(); | ||||||
|  |  | ||||||
|    const diff = process.hrtime(t); |    const diff = process.hrtime(t); | ||||||
|    const NS_PER_SEC = 1e9; |    const NS_PER_SEC = 1e9; | ||||||
|    await Logging.waitForSetup(); |  | ||||||
|    const ns = diff[0] * NS_PER_SEC + diff[1]; |    const ns = diff[0] * NS_PER_SEC + diff[1]; | ||||||
|    console.log(`Benchmark took ${ns / 1000000}ms for ${count} messages with a size of ${message_size} characters`); |    console.log( | ||||||
|    console.log(`This is equal to ${(ns / 1000000) / count} ms per message`) |       `Benchmark took ${ | ||||||
|  |          ns / 1000000 | ||||||
|  |       }ms for ${count} messages with a size of ${message_size} characters` | ||||||
|  |    ); | ||||||
|  |    console.log(`This is equal to ${ns / 1000000 / count} ms per message`); | ||||||
| } | } | ||||||
|  |  | ||||||
| Logging.waitForSetup().then(async () => { | const benchTimer = Logging.time("benchmark"); | ||||||
|    console.log("Large data benchmark:") | Promise.resolve().then(async () => { | ||||||
|    await benchmark(7000, 50000); |    console.log("Large data benchmark:"); | ||||||
|  |    await benchmark(70000, 50000); | ||||||
|  |  | ||||||
|    console.log("Realdata data benchmark:") |    console.log("Realdata data benchmark:"); | ||||||
|    await benchmark(100000, 100) |    await benchmark(100000, 100); | ||||||
|  |    benchTimer.end(); | ||||||
|  |  | ||||||
|  |    const timer = Logging.time("Test Timer"); | ||||||
|  |    setTimeout(() => timer.end(), 1000); | ||||||
| }); | }); | ||||||
							
								
								
									
										11
									
								
								tsconfig.esm.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								tsconfig.esm.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | { | ||||||
|  |    "extends": "./tsconfig.json", | ||||||
|  |    "compilerOptions": { | ||||||
|  |       "module": "ESNext", | ||||||
|  |       "target": "ES2017", | ||||||
|  |       "moduleResolution": "node", | ||||||
|  |       "outDir": "esm" | ||||||
|  |    }, | ||||||
|  |    "exclude": ["node_modules"], | ||||||
|  |    "include": ["src"] | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user
	