1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-26 18:58:21 +02:00

Desktop, Cli: Improved log formatting and allow saving last lines of log to memory

This commit is contained in:
Laurent Cozic 2024-04-26 15:08:37 +01:00
parent 993fbfb93f
commit 74bc9b36aa
2 changed files with 126 additions and 9 deletions

View File

@ -0,0 +1,89 @@
import Logger, { LogLevel, TargetType } from './Logger';
import { WriteFileOptions, appendFile, mkdirp, readFile, remove } from 'fs-extra';
Logger.fsDriver_ = {
appendFile: async (path, content, encoding) => {
return await appendFile(path, content, encoding as WriteFileOptions);
},
};
const testDirPath = `${__dirname}/LoggerTests`;
const logPath = () => {
return `${testDirPath}/log.txt`;
};
const getLogContent = async () => {
return readFile(logPath(), 'utf-8');
};
const createLogger = () => {
const logger = new Logger();
logger.addTarget(TargetType.File, {
prefix: 'testing',
path: logPath(),
level: LogLevel.Debug,
});
return logger;
};
describe('Logger', () => {
beforeEach(async () => {
await mkdirp(testDirPath);
});
afterEach(async () => {
await remove(testDirPath);
});
it('should log to file', async () => {
jest.useFakeTimers().setSystemTime(new Date('2020-01-01'));
const logger = createLogger();
logger.debug('one');
logger.warn('two');
logger.error('three');
await logger.waitForFileWritesToComplete_();
expect(await getLogContent()).toBe([
'2020-01-01 00:00:00: testing: one',
'2020-01-01 00:00:00: testing: [warn] two',
'2020-01-01 00:00:00: testing: [error] three',
'',
].join('\n'));
// Shouldn't have kept any line, since keptLineCount was not set
expect(logger.keptLines).toEqual([]);
jest.useRealTimers();
});
it('should keep the last lines', async () => {
jest.useFakeTimers().setSystemTime(new Date('2020-01-01'));
const logger = createLogger();
logger.keptLineCount = 2;
logger.info('one');
logger.info('two');
logger.info('three');
await logger.waitForFileWritesToComplete_();
expect(await getLogContent()).toBe([
'2020-01-01 00:00:00: testing: one',
'2020-01-01 00:00:00: testing: two',
'2020-01-01 00:00:00: testing: three',
'',
].join('\n'));
expect(logger.keptLines).toEqual([
'2020-01-01 00:00:00: testing: two',
'2020-01-01 00:00:00: testing: three',
]);
jest.useRealTimers();
});
});

View File

@ -78,6 +78,8 @@ class Logger {
private level_: LogLevel = LogLevel.Info;
private lastDbCleanup_: number = Date.now();
private enabled_ = true;
private keptLineCount_ = 0;
private keptLines_: string[] = [];
public static fsDriver() {
if (!Logger.fsDriver_) Logger.fsDriver_ = dummyFsDriver;
@ -92,6 +94,18 @@ class Logger {
this.enabled_ = v;
}
public get keptLineCount() {
return this.keptLineCount_;
}
public set keptLineCount(v: number) {
this.keptLineCount_ = v;
}
public get keptLines() {
return this.keptLines_;
}
public status(): string {
const output: string[] = [];
output.push(`Enabled: ${this.enabled}`);
@ -252,13 +266,23 @@ class Logger {
return this.level();
}
// eslint-disable-next-line @typescript-eslint/no-explicit-any -- Old code before rule was applied
public log(level: LogLevel, prefix: string | null, ...object: any[]) {
private logInfoToString(level: LogLevel, prefix: string | null, ...object: unknown[]) {
const timestamp = moment().format('YYYY-MM-DD HH:mm:ss');
const line = [timestamp];
if (prefix) line.push(prefix);
const levelString = [LogLevel.Error, LogLevel.Warn].includes(level) ? `[${Logger.levelIdToString(level)}] ` : '';
line.push((levelString ? levelString : '') + this.objectsToString(...object));
return `${line.join(': ')}`;
}
public log(level: LogLevel, prefix: string | null, ...object: unknown[]) {
if (!this.targets_.length || !this.enabled) return;
let logLine = '';
for (let i = 0; i < this.targets_.length; i++) {
const target = this.targets_[i];
const targetPrefix = prefix ? prefix : target.prefix;
const targetPrefix = prefix ? prefix : (target.prefix || '');
if (this.targetLevel(target) < level) continue;
@ -285,15 +309,12 @@ class Logger {
} else {
const prefixItems = [moment().format('HH:mm:ss')];
if (targetPrefix) prefixItems.push(targetPrefix);
items = [`${prefixItems.join(': ')}:`].concat(...object);
items = [`${prefixItems.join(': ')}:` as unknown].concat(...object);
}
consoleObj[fn](...items);
} else if (target.type === 'file') {
const timestamp = moment().format('YYYY-MM-DD HH:mm:ss');
const line = [timestamp];
if (targetPrefix) line.push(targetPrefix);
line.push(this.objectsToString(...object));
logLine = this.logInfoToString(level, targetPrefix, ...object);
// Write to file using a mutex so that log entries appear in the
// correct order (otherwise, since the async call is not awaited
@ -307,7 +328,7 @@ class Logger {
/* eslint-disable-next-line promise/prefer-await-to-then, @typescript-eslint/ban-types -- Old code before rule was applied, Old code before rule was applied */
writeToFileMutex_.acquire().then((r: Function) => {
release = r;
return Logger.fsDriver().appendFile(target.path as string, `${line.join(': ')}\n`, 'utf8');
return Logger.fsDriver().appendFile(target.path as string, `${logLine}\n`, 'utf8');
// eslint-disable-next-line promise/prefer-await-to-then, @typescript-eslint/no-explicit-any -- Old code before rule was applied, Old code before rule was applied
}).catch((error: any) => {
console.error('Cannot write to log file:', error);
@ -340,6 +361,13 @@ class Logger {
target.database.transactionExecBatch(queries);
}
}
if (this.keptLineCount) {
if (!logLine) logLine = this.logInfoToString(level, prefix, ...object);
this.keptLines_.push(logLine);
const toDelete = this.keptLines_.length - this.keptLineCount;
if (toDelete >= 0) this.keptLines_.splice(0, toDelete);
}
}
// For tests