2019-01-16 12:00:59 -06:00
|
|
|
import * as archiver from 'archiver';
|
2018-12-03 11:46:47 -06:00
|
|
|
import { app, BrowserWindow, dialog, shell } from 'electron';
|
2019-01-16 12:00:59 -06:00
|
|
|
import * as electron from 'electron';
|
2018-12-03 11:46:47 -06:00
|
|
|
import * as fs from 'fs';
|
2019-01-16 12:00:59 -06:00
|
|
|
import * as path from 'path';
|
2018-12-03 11:46:47 -06:00
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
import { ILogs } from '../common/api-interface';
|
2019-08-20 04:00:13 -05:00
|
|
|
import { isLinux, isMac } from '../common/env';
|
2018-12-03 11:46:47 -06:00
|
|
|
import { i18n } from '../common/i18n';
|
2019-05-16 08:30:00 -05:00
|
|
|
import { logger } from '../common/logger';
|
2018-12-03 11:46:47 -06:00
|
|
|
|
2019-01-16 12:00:59 -06:00
|
|
|
/**
|
|
|
|
* Archives files in the source directory
|
|
|
|
* that matches the given file extension
|
|
|
|
*
|
|
|
|
* @param source {String} source path
|
|
|
|
* @param destination {String} destination path
|
|
|
|
* @param fileExtensions {Array} array of file ext
|
|
|
|
* @return {Promise<void>}
|
|
|
|
*/
|
2019-11-27 07:04:58 -06:00
|
|
|
const generateArchiveForDirectory = (source: string, destination: string, fileExtensions: string[], retrievedLogs: ILogs[]): Promise<void> => {
|
2019-01-16 12:00:59 -06:00
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
2019-05-16 08:30:00 -05:00
|
|
|
logger.info(`reports-handler: generating archive for directory ${source}`);
|
2019-01-16 12:00:59 -06:00
|
|
|
const output = fs.createWriteStream(destination);
|
|
|
|
const archive = archiver('zip', { zlib: { level: 9 } });
|
2019-11-27 07:04:58 -06:00
|
|
|
const filesForCleanup: string[] = [];
|
2019-01-16 12:00:59 -06:00
|
|
|
|
|
|
|
output.on('close', () => {
|
2019-11-27 07:04:58 -06:00
|
|
|
for (const file of filesForCleanup) {
|
|
|
|
if (fs.existsSync( file )) {
|
|
|
|
fs.unlinkSync(file);
|
|
|
|
}
|
|
|
|
}
|
2019-05-16 08:30:00 -05:00
|
|
|
logger.info(`reports-handler: generated archive for directory ${source}`);
|
2019-01-16 12:00:59 -06:00
|
|
|
return resolve();
|
|
|
|
});
|
|
|
|
|
2019-05-16 08:30:00 -05:00
|
|
|
archive.on('error', (err: Error) => {
|
2019-11-27 07:04:58 -06:00
|
|
|
for (const file of filesForCleanup) {
|
|
|
|
if (fs.existsSync( file )) {
|
|
|
|
fs.unlinkSync(file);
|
|
|
|
}
|
|
|
|
}
|
2019-05-16 08:30:00 -05:00
|
|
|
logger.error(`reports-handler: error archiving directory for ${source} with error ${err}`);
|
2019-01-16 12:00:59 -06:00
|
|
|
return reject(err);
|
|
|
|
});
|
|
|
|
|
|
|
|
archive.pipe(output);
|
|
|
|
|
|
|
|
const files = fs.readdirSync(source);
|
|
|
|
files
|
|
|
|
.filter((file) => fileExtensions.indexOf(path.extname(file)) !== -1)
|
|
|
|
.forEach((file) => {
|
|
|
|
switch (path.extname(file)) {
|
|
|
|
case '.log':
|
|
|
|
archive.file(source + '/' + file, { name: 'logs/' + file });
|
|
|
|
break;
|
|
|
|
case '.dmp':
|
|
|
|
case '.txt': // on Windows .txt files will be created as part of crash dump
|
|
|
|
archive.file(source + '/' + file, { name: 'crashes/' + file });
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
for (const logs of retrievedLogs) {
|
|
|
|
for (const logFile of logs.logFiles) {
|
|
|
|
const file = path.join( source, logFile.filename );
|
|
|
|
fs.writeFileSync(file, logFile.contents );
|
|
|
|
archive.file(file, { name: 'logs/' + logFile.filename });
|
|
|
|
filesForCleanup.push(file);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-16 12:00:59 -06:00
|
|
|
archive.finalize();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
interface ILogRetriever {
|
|
|
|
sender: Electron.WebContents;
|
|
|
|
logName: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const logRetrievers: ILogRetriever[] = [];
|
|
|
|
const receivedLogs: ILogs[] = [];
|
|
|
|
|
|
|
|
export const registerLogRetriever = ( sender: Electron.WebContents, logName: string): void => {
|
|
|
|
logRetrievers.push( { sender, logName } );
|
|
|
|
};
|
|
|
|
|
|
|
|
export const collectLogs = (): void => {
|
|
|
|
receivedLogs.length = 0;
|
|
|
|
for (const logRetriever of logRetrievers ) {
|
|
|
|
logRetriever.sender.send('collect-logs', logRetriever.logName );
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-01-16 12:00:59 -06:00
|
|
|
/**
|
|
|
|
* Compress and export logs stored under system log directory
|
|
|
|
*
|
|
|
|
* MacOS - /Library/Logs/Symphony/
|
|
|
|
* Windows - AppData\Roaming\Symphony\logs
|
|
|
|
*/
|
2019-11-27 07:04:58 -06:00
|
|
|
export const packageLogs = (retrievedLogs: ILogs[]): void => {
|
2018-12-03 11:46:47 -06:00
|
|
|
const FILE_EXTENSIONS = [ '.log' ];
|
|
|
|
const MAC_LOGS_PATH = '/Library/Logs/Symphony/';
|
2019-08-20 04:00:13 -05:00
|
|
|
const LINUX_LOGS_PATH = '/.config/Symphony/';
|
2019-10-30 10:28:25 -05:00
|
|
|
const WINDOWS_LOGS_PATH = '\\AppData\\Local\\Symphony\\Symphony\\logs';
|
2018-12-03 11:46:47 -06:00
|
|
|
|
2019-08-20 04:00:13 -05:00
|
|
|
const logsPath = isMac ? MAC_LOGS_PATH : isLinux ? LINUX_LOGS_PATH : WINDOWS_LOGS_PATH;
|
2018-12-03 11:46:47 -06:00
|
|
|
const source = app.getPath('home') + logsPath;
|
|
|
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
|
|
|
|
|
|
|
if (!fs.existsSync(source) && focusedWindow && !focusedWindow.isDestroyed()) {
|
2019-05-16 08:30:00 -05:00
|
|
|
logger.error(`reports-handler: Can't find any logs to share!`);
|
2018-12-03 11:46:47 -06:00
|
|
|
dialog.showMessageBox(focusedWindow, {
|
2019-05-16 08:30:00 -05:00
|
|
|
message: i18n.t(`Can't find any logs to share!`)(),
|
2018-12-08 09:47:09 -06:00
|
|
|
title: i18n.t('Failed!')(),
|
2018-12-03 11:46:47 -06:00
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
2019-08-20 04:00:13 -05:00
|
|
|
const destPath = (isMac || isLinux) ? '/logs_symphony_' : '\\logs_symphony_';
|
2018-12-03 11:46:47 -06:00
|
|
|
const timestamp = new Date().getTime();
|
|
|
|
const destination = app.getPath('downloads') + destPath + timestamp + '.zip';
|
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
generateArchiveForDirectory(source, destination, FILE_EXTENSIONS, retrievedLogs)
|
2018-12-03 11:46:47 -06:00
|
|
|
.then(() => {
|
|
|
|
shell.showItemInFolder(destination);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
if (focusedWindow && !focusedWindow.isDestroyed()) {
|
2019-05-16 08:30:00 -05:00
|
|
|
logger.error(`reports-handler: Can't share logs due to error ${err}`);
|
2018-12-03 11:46:47 -06:00
|
|
|
dialog.showMessageBox(focusedWindow, {
|
2018-12-08 09:47:09 -06:00
|
|
|
message: `${i18n.t('Unable to generate logs due to ')()} ${err}`,
|
|
|
|
title: i18n.t('Failed!')(),
|
2018-12-03 11:46:47 -06:00
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
export const finalizeLogExports = (logs: ILogs) => {
|
|
|
|
receivedLogs.push(logs);
|
|
|
|
|
|
|
|
let allReceived = true;
|
|
|
|
for (const logRetriever of logRetrievers ) {
|
|
|
|
let found = false;
|
|
|
|
for (const log of receivedLogs) {
|
|
|
|
if (log.logName === logRetriever.logName) {
|
|
|
|
found = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!found) {
|
|
|
|
allReceived = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (allReceived) {
|
|
|
|
packageLogs(receivedLogs);
|
|
|
|
receivedLogs.length = 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export const exportLogs = (): void => {
|
|
|
|
if ( logRetrievers.length > 0) {
|
|
|
|
collectLogs();
|
|
|
|
} else {
|
|
|
|
packageLogs([]);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-01-16 12:00:59 -06:00
|
|
|
/**
|
|
|
|
* Compress and export crash dump stored under system crashes directory
|
|
|
|
*/
|
2018-12-03 11:46:47 -06:00
|
|
|
export const exportCrashDumps = (): void => {
|
|
|
|
const FILE_EXTENSIONS = isMac ? [ '.dmp' ] : [ '.dmp', '.txt' ];
|
|
|
|
const crashesDirectory = (electron.crashReporter as any).getCrashesDirectory();
|
|
|
|
const source = isMac ? crashesDirectory + '/completed' : crashesDirectory;
|
|
|
|
const focusedWindow = BrowserWindow.getFocusedWindow();
|
|
|
|
|
|
|
|
if (!fs.existsSync(source) || fs.readdirSync(source).length === 0 && focusedWindow && !focusedWindow.isDestroyed()) {
|
|
|
|
electron.dialog.showMessageBox(focusedWindow as BrowserWindow, {
|
2018-12-08 09:47:09 -06:00
|
|
|
message: i18n.t('No crashes available to share')(),
|
|
|
|
title: i18n.t('Failed!')(),
|
2018-12-03 11:46:47 -06:00
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-08-20 04:00:13 -05:00
|
|
|
const destPath = (isMac || isLinux) ? '/crashes_symphony_' : '\\crashes_symphony_';
|
2018-12-03 11:46:47 -06:00
|
|
|
const timestamp = new Date().getTime();
|
|
|
|
|
|
|
|
const destination = electron.app.getPath('downloads') + destPath + timestamp + '.zip';
|
|
|
|
|
2019-11-27 07:04:58 -06:00
|
|
|
generateArchiveForDirectory(source, destination, FILE_EXTENSIONS, [])
|
2018-12-03 11:46:47 -06:00
|
|
|
.then(() => {
|
|
|
|
electron.shell.showItemInFolder(destination);
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
if (focusedWindow && !focusedWindow.isDestroyed()) {
|
|
|
|
electron.dialog.showMessageBox(focusedWindow, {
|
2018-12-08 09:47:09 -06:00
|
|
|
message: `${i18n.t('Unable to generate crash reports due to ')()} ${err}`,
|
|
|
|
title: i18n.t('Failed!')(),
|
2018-12-03 11:46:47 -06:00
|
|
|
type: 'error',
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2019-03-19 05:52:39 -05:00
|
|
|
};
|