ultrawidify/src/ext/lib/Logger.js

312 lines
9.4 KiB
JavaScript
Raw Normal View History

2019-09-03 21:30:18 +02:00
import Debug from '../conf/Debug';
import currentBrowser from '../conf/BrowserDetect';
2019-07-16 20:59:12 +02:00
class Logger {
constructor(conf) {
}
static saveConfig(conf) {
if (currentBrowser.firefox || currentBrowser.edge) {
return browser.storage.local.set( {'uwLogger': JSON.stringify(conf)});
} else if (currentBrowser.chrome) {
return chrome.storage.local.set( {'uwLogger': JSON.stringify(conf)});
}
}
static syncConfig(callback) {
const br = currentBrowser.firefox ? browser : chrome;
br.storage.onChanged.addListener( (changes, area) => {
callback(JSON.parse(changes.uwLogger.newValue));
});
}
static async getConfig() {
let ret;
if (currentBrowser.firefox) {
ret = await browser.storage.local.get('uwLogger');
} else if (currentBrowser.chrome) {
ret = await new Promise( (resolve, reject) => {
chrome.storage.local.get('uwLogger', (res) => resolve(res));
});
} else if (currentBrowser.edge) {
ret = await new Promise( (resolve, reject) => {
browser.storage.local.get('uwLogger', (res) => resolve(res));
});
}
if (Debug.debug && Debug.debugStorage) {
try {
console.log("[Logger::getSaved] Got settings:", JSON.parse(ret.uwLogger));
} catch (e) {
console.log("[Logger::getSaved] No settings.")
}
}
try {
return JSON.parse(ret.uwLogger);
} catch(e) {
return {logToFile: false, logToConsole: false, consoleOptions: {}, fileOptions: {}};
}
}
2020-01-27 22:32:38 +01:00
async init(conf) {
if (conf && process.env.CHANNEL === 'dev') {
this.conf = conf;
} else {
this.conf = await Logger.getConfig();
}
if (this.conf.consoleOptions === undefined) {
this.conf.consoleOptions = {};
}
if (!this.conf.fileOptions === undefined) {
this.conf.fileOptions = {};
}
// this is the only property that always gets passed via conf
// and doesn't get ignored even if the rest of the conf gets
// loaded from browser storage
this.isBackgroundPage = !!conf.isBackgroundPage;
this.history = [];
this.globalHistory = {};
this.startTime = performance.now();
this.temp_disable = false;
this.stopTime = conf.timeout ? performance.now() + (conf.timeout * 1000) : undefined;
const ths = this;
const br = currentBrowser.firefox ? browser : chrome;
br.storage.onChanged.addListener( (changes, area) => {
if (Debug.debug && Debug.debugStorage) {
console.log("[Logger::<storage/on change>] Settings have been changed outside of here. Updating active settings. Changes:", changes, "storage area:", area);
if (changes['uwLogger'] && changes['uwLogger'].newValue) {
console.log("[Logger::<storage/on change>] new settings object:", JSON.parse(changes.uwLogger.newValue));
}
}
if(changes['uwLogger'] && changes['uwLogger'].newValue) {
ths.conf = JSON.parse(changes.uwLogger.newValue);
}
});
}
2019-07-16 20:59:12 +02:00
clear() {
this.log = [];
this.startTime = performance.now();
2020-01-06 21:37:40 +01:00
this.stopTime = this.conf.timeout ? performance.now() + (this.conf.timeout * 1000) : undefined;
2019-07-16 20:59:12 +02:00
}
setConf(conf) {
this.conf = conf; // effective immediately
// also persist settings:
Logger.saveConfig(conf);
2019-07-16 20:59:12 +02:00
}
async getSaved() {
return Logger.getSaved();
}
2019-07-16 20:59:12 +02:00
// allow syncing of start times between bg and page scripts.
// may result in negative times in the log file, but that doesn't
// really matter
getStartTime() {
return this.startTime;
}
setStartTime(startTime) {
if (startTime) {
this.startTime = startTime;
} else {
this.startTime = performance.now();
}
}
2020-01-06 21:37:40 +01:00
// getLogFileString() {
// let logfileStr = '';
// let logTs = ''; // number of seconds since extension started on a given page¸
// for (let i = 0; i < this.history.length; i++) {
// logTs = ((this.history[i].ts - Math.floor(this.performance.now)) / 3).toFixed(3);
// logfileStr = `${logfileStr}[@${logTs}] -- ${this.history[i].message}\n`
// }
// return logfileStr;
// }
getFileLogJSONString() {
return {
site: window && window.location,
log: JSON.toString(this.history),
2019-07-16 20:59:12 +02:00
}
}
pause() {
this.temp_disable = true;
}
resume() {
this.temp_disable = false;
}
canLog(component) {
return this.conf.allowLogging && (this.canLogFile(component) || this.canLogConsole(component));
}
canLogFile(component) {
if (!this.conf.fileOptions.enabled || this.temp_disable) {
return false;
}
2020-01-06 21:37:40 +01:00
if (performance.now() > this.stopTime) {
this.conf.allowLogging = false;
2020-01-06 21:37:40 +01:00
return false;
}
2019-11-04 22:14:41 +01:00
if (Array.isArray(component) && component.length ) {
for (const c of component) {
if (this.conf.fileOptions[c]) {
return this.conf.fileOptions[c];
}
}
2019-11-04 22:14:41 +01:00
} else {
return this.conf.fileOptions[component];
}
}
canLogConsole(component) {
if (!this.conf.consoleOptions.enabled || this.temp_disable) {
return false;
}
2020-01-06 21:37:40 +01:00
if (performance.now() > this.stopTime) {
this.conf.allowLogging = false;
2020-01-06 21:37:40 +01:00
return false;
}
2019-08-25 22:00:57 +02:00
if (Array.isArray(component) && component.length) {
2019-11-04 22:14:41 +01:00
for (const c of component) {
if (this.conf.consoleOptions[c]) {
return this.conf.consoleOptions[c];
}
}
2019-11-04 22:14:41 +01:00
} else {
2020-01-15 21:09:12 +01:00
return this.conf.consoleOptions[component] !== undefined ? this.conf.consoleOptions[component] : this.conf.logAll;
2019-11-04 22:14:41 +01:00
}
2020-01-15 21:09:12 +01:00
return this.conf.logAll;
}
2019-07-16 20:59:12 +02:00
// level is unused as of now, but this may change in the future
// levels: 'info', 'warn', 'error'
2019-07-16 20:59:12 +02:00
log(level, component, ...message) {
if (!this.conf) {
return;
}
2020-01-06 21:37:40 +01:00
const error = new Error();
2019-07-16 20:59:12 +02:00
if (this.conf.logToFile) {
if (this.canLogFile(component)) {
2019-07-16 20:59:12 +02:00
let ts = performance.now();
if (ts <= this.history[this.history.length - 1]) {
ts = this.history[this.history.length - 1] + 0.00001;
}
this.history.push({
ts: ts,
message: JSON.stringify(message),
2020-01-06 21:37:40 +01:00
stack: error.stack,
2019-07-16 20:59:12 +02:00
})
}
}
if (this.conf.logToConsole) {
if (this.canLogConsole(component)) {
2020-01-06 21:37:40 +01:00
console.log(...message, error.stack);
2019-07-16 20:59:12 +02:00
}
}
}
// leaves a noticeable mark in the file log at the time it got triggered, as well as
// at the intervals of 1s and .5s before the trigger moment
cahen() {
if (this.conf.logToFile) {
let ts = performance.now();
let secondMark = ts - 1000;
let halfSecondMark = ts - 500;
let i = this.history.length();
// correct ts _after_ secondMark and halfSecondMark were determined
if (ts <= this.history[this.history.length - 1]) {
ts = this.history[this.history.length - 1] + 0.00001;
}
this.history.push({
ts: ts,
message: "-------------------------------------- CAHEN --------------------------------------"
});
// find the spot for the half second mark. In this case, we don't really particularly care whether timestamps
// are duped due to cahen warnings
while (this.history[i--].ts > halfSecondMark) {}
this.history.push({
ts: halfSecondMark,
message: "-------------------------------------- 0.5s to CAHEN --------------------------------------"
});
// repeat for one second warning
while (this.history[i--].ts > secondMark) {}
this.history.push({
ts: secondMark,
message: "-------------------------------------- 1s to CAHEN --------------------------------------"
});
}
}
2020-01-21 00:41:06 +01:00
addLogFromPage(host, tabId, frameId, pageHistory) {
if (! this.globalHistory[host]) {
this.globalHistory[host] = {};
}
if (! this.globalHistory[tabId || 'tab']) {
this.globalHistory[host][tabId || 'tab'] = {};
}
if (!this.globalHistory[frameId || 'top']) {
this.globalHistory[host][tabId || 'tab'][frameId || 'top'] = pageHistory;
} else {
this.globalHistory[host][tabId || 'tab'][frameId || 'top'].push(...pageHistory);
}
}
// export log file — only works on background page
async exportLogToFile() {
const exportObject = {'pageLogs': JSON.parse(JSON.stringify({...this.globalHistory}))};
exportObject['logger-settings'] = this.conf.fileOptions;
exportObject['backgroundLog'] = JSON.parse(JSON.stringify(this.history));
exportObject['popupLog'] = 'NOT IMPLEMENTED';
const blob = new Blob([JSON.stringify(exportObject)], {type: 'application/json'});
const fileUrl = URL.createObjectURL(blob);
2019-07-16 20:59:12 +02:00
2020-01-21 00:41:06 +01:00
try {
if (BrowserDetect.firefox) {
await browser.permissions.request({permissions: ['downloads']});
browser.downloads.download({saveAs: true, filename: 'extension-log.json', url: fileUrl});
} else if (BrowserDetect.chrome) {
const ths = this;
chrome.permissions.request(
{permissions: ['downloads']},
(granted) => {
if (granted) {
chrome.downloads.download({saveAs: true, filename: 'extension-log.json', url: fileUrl});
} else {
ths.downloadPermissionError = true
}
}
)
}
this.globalHistory = {};
this.history = [];
} catch (e) {
this.downloadPermissionError = true;
}
}
// used for instances where logging is limited to a single page and is timed
addLogAndExport(host, pageHistory) {
this.globalHistory = {};
this.globalHistory[host || 'no-host-provided'] = pageHistory;
this.exportLogToFile();
}
2019-07-16 20:59:12 +02:00
}
export default Logger;