// Copyright 2017-2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only // NOTE: Temporarily allow `then` until we convert the entire file to `async` / `await`: /* eslint-disable more/no-then */ /* eslint-disable no-console */ import { join } from 'path'; import { readdirSync, readFile, unlinkSync, writeFileSync } from 'fs'; import { BrowserWindow, app, ipcMain as ipc } from 'electron'; import pinoms from 'pino-multi-stream'; import pino from 'pino'; import * as mkdirp from 'mkdirp'; import { compact, filter, flatten, map, pick, sortBy } from 'lodash'; import readFirstLine from 'firstline'; import { read as readLastLines } from 'read-last-lines'; import rimraf from 'rimraf'; import { createStream } from 'rotating-file-stream'; import type { LoggerType } from '../types/Logging'; import * as log from './log'; import { Environment, getEnvironment } from '../environment'; import { FetchLogIpcData, LogEntryType, LogLevel, cleanArgs, getLogLevelString, isLogEntry, } from './shared'; declare global { // We want to extend `Console`, so we need an interface. // eslint-disable-next-line no-restricted-syntax interface Console { _log: typeof console.log; _warn: typeof console.warn; _error: typeof console.error; } } let globalLogger: undefined | pino.Logger; let shouldRestart = false; const isRunningFromConsole = Boolean(process.stdout.isTTY) || getEnvironment() === Environment.Test || getEnvironment() === Environment.TestLib; export async function initialize( getMainWindow: () => undefined | BrowserWindow ): Promise { if (globalLogger) { throw new Error('Already called initialize!'); } const basePath = app.getPath('userData'); const logPath = join(basePath, 'logs'); mkdirp.sync(logPath); try { await cleanupLogs(logPath); } catch (error) { const errorString = `Failed to clean logs; deleting all. Error: ${error.stack}`; console.error(errorString); await deleteAllLogs(logPath); mkdirp.sync(logPath); // If we want this log entry to persist on disk, we need to wait until we've // set up our logging infrastructure. setTimeout(() => { console.error(errorString); }, 500); } const logFile = join(logPath, 'main.log'); const stream = createStream(logFile, { interval: '1d', rotate: 3, }); const onClose = () => { globalLogger = undefined; if (shouldRestart) { initialize(getMainWindow); } }; stream.on('close', onClose); stream.on('error', onClose); const streams: pinoms.Streams = []; streams.push({ stream }); if (isRunningFromConsole) { streams.push({ level: 'debug' as const, stream: process.stdout, }); } const logger = pinoms({ streams, timestamp: pino.stdTimeFunctions.isoTime, }); ipc.on('fetch-log', async event => { const mainWindow = getMainWindow(); if (!mainWindow) { logger.info('Logs were requested, but the main window is missing'); return; } let data: FetchLogIpcData; try { const [logEntries, rest] = await Promise.all([ fetchLogs(logPath), fetchAdditionalLogData(mainWindow), ]); data = { logEntries, ...rest, }; } catch (error) { logger.error(`Problem loading log data: ${error.stack}`); return; } try { event.sender.send('fetched-log', data); } catch (err: unknown) { // NOTE(evanhahn): We don't want to send a message to a window that's closed. // I wanted to use `event.sender.isDestroyed()` but that seems to fail. // Instead, we attempt the send and catch the failure as best we can. const hasUserClosedWindow = isProbablyObjectHasBeenDestroyedError(err); if (hasUserClosedWindow) { logger.info('Logs were requested, but it seems the window was closed'); } else { logger.error( 'Problem replying with fetched logs', err instanceof Error && err.stack ? err.stack : err ); } } }); ipc.on('delete-all-logs', async event => { // Restart logging when the streams will close shouldRestart = true; try { await deleteAllLogs(logPath); } catch (error) { logger.error(`Problem deleting all logs: ${error.stack}`); } event.sender.send('delete-all-logs-complete'); }); globalLogger = logger; return log; } async function deleteAllLogs(logPath: string): Promise { return new Promise((resolve, reject) => { rimraf( logPath, { disableGlob: true, }, error => { if (error) { return reject(error); } return resolve(); } ); }); } async function cleanupLogs(logPath: string) { const now = new Date(); const earliestDate = new Date( Date.UTC(now.getUTCFullYear(), now.getUTCMonth(), now.getUTCDate() - 3) ); try { const remaining = await eliminateOutOfDateFiles(logPath, earliestDate); const files = filter(remaining, file => !file.start && file.end); if (!files.length) { return; } await eliminateOldEntries(files, earliestDate); } catch (error) { console.error( 'Error cleaning logs; deleting and starting over from scratch.', error.stack ); // delete and re-create the log directory await deleteAllLogs(logPath); mkdirp.sync(logPath); } } // Exported for testing only. export function isLineAfterDate(line: string, date: Readonly): boolean { if (!line) { return false; } try { const data = JSON.parse(line); return new Date(data.time).getTime() > date.getTime(); } catch (e) { console.log('error parsing log line', e.stack, line); return false; } } // Exported for testing only. export function eliminateOutOfDateFiles( logPath: string, date: Readonly ): Promise< Array<{ path: string; start: boolean; end: boolean; }> > { const files = readdirSync(logPath); const paths = files.map(file => join(logPath, file)); return Promise.all( map(paths, target => Promise.all([readFirstLine(target), readLastLines(target, 2)]).then( results => { const start = results[0]; const end = results[1].split('\n'); const file = { path: target, start: isLineAfterDate(start, date), end: isLineAfterDate(end[end.length - 1], date) || isLineAfterDate(end[end.length - 2], date), }; if (!file.start && !file.end) { unlinkSync(file.path); } return file; } ) ) ); } // Exported for testing only. export async function eliminateOldEntries( files: ReadonlyArray<{ path: string }>, date: Readonly ): Promise { await Promise.all( map(files, file => fetchLog(file.path).then(lines => { const recent = filter(lines, line => new Date(line.time) >= date); const text = map(recent, line => JSON.stringify(line)).join('\n'); return writeFileSync(file.path, `${text}\n`); }) ) ); } // Exported for testing only. export function fetchLog(logFile: string): Promise> { return new Promise((resolve, reject) => { readFile(logFile, { encoding: 'utf8' }, (err, text) => { if (err) { return reject(err); } const lines = compact(text.split('\n')); const data = compact( lines.map(line => { try { const result = pick(JSON.parse(line), ['level', 'time', 'msg']); return isLogEntry(result) ? result : null; } catch (e) { return null; } }) ); return resolve(data); }); }); } // Exported for testing only. export function fetchLogs(logPath: string): Promise> { const files = readdirSync(logPath); const paths = files.map(file => join(logPath, file)); // creating a manual log entry for the final log result const fileListEntry: LogEntryType = { level: LogLevel.Info, time: new Date().toISOString(), msg: `Loaded this list of log files from logPath: ${files.join(', ')}`, }; return Promise.all(paths.map(fetchLog)).then(results => { const data = flatten(results); data.push(fileListEntry); return sortBy(data, logEntry => logEntry.time); }); } export const fetchAdditionalLogData = ( mainWindow: BrowserWindow ): Promise> => new Promise(resolve => { mainWindow.webContents.send('additional-log-data-request'); ipc.once('additional-log-data-response', (_event, data) => { resolve(data); }); }); function logAtLevel(level: LogLevel, ...args: ReadonlyArray) { if (globalLogger) { const levelString = getLogLevelString(level); globalLogger[levelString](cleanArgs(args)); } else if (isRunningFromConsole && !process.stdout.destroyed) { console._log(...args); } } function isProbablyObjectHasBeenDestroyedError(err: unknown): boolean { return err instanceof Error && err.message === 'Object has been destroyed'; } // This blows up using mocha --watch, so we ensure it is run just once if (!console._log) { log.setLogAtLevel(logAtLevel); console._log = console.log; console.log = log.info; console._error = console.error; console.error = log.error; console._warn = console.warn; console.warn = log.warn; }