238 lines
6.9 KiB
JavaScript
238 lines
6.9 KiB
JavaScript
import Debug from '../conf/Debug';
|
||
import currentBrowser from '../conf/BrowserDetect';
|
||
|
||
class Logger {
|
||
constructor(conf) {
|
||
this.initLogger();
|
||
if (conf) {
|
||
this.conf = conf;
|
||
}
|
||
if (this.conf.consoleOptions === undefined) {
|
||
this.conf.consoleOptions = {};
|
||
}
|
||
if (!this.conf.fileOptions === undefined) {
|
||
this.conf.fileOptions = {};
|
||
}
|
||
this.history = [];
|
||
this.startTime = performance.now();
|
||
this.temp_disable = false;
|
||
this.stopTime = conf.timeout ? performance.now() + (conf.timeout * 1000) : undefined;
|
||
}
|
||
|
||
initLogger() {
|
||
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);
|
||
}
|
||
});
|
||
}
|
||
|
||
async init() {
|
||
if (!this.conf) {
|
||
this.conf = await this.getSaved();
|
||
}
|
||
}
|
||
|
||
clear() {
|
||
this.log = [];
|
||
this.startTime = performance.now();
|
||
this.stopTime = this.conf.timeout ? performance.now() + (this.conf.timeout * 1000) : undefined;
|
||
}
|
||
|
||
setConf(conf) {
|
||
this.conf = conf; // effective immediately
|
||
// also persist settings:
|
||
if (currentBrowser.firefox || currentBrowser.edge) {
|
||
return browser.storage.local.set( {'uwLogger': JSON.stringify(this.conf)});
|
||
} else if (currentBrowser.chrome) {
|
||
return chrome.storage.local.set( {'uwLogger': JSON.stringify(this.logger)});
|
||
}
|
||
}
|
||
|
||
async getSaved() {
|
||
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: {}};
|
||
}
|
||
}
|
||
|
||
|
||
// 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();
|
||
}
|
||
}
|
||
|
||
// 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),
|
||
}
|
||
}
|
||
|
||
pause() {
|
||
this.temp_disable = true;
|
||
}
|
||
resume() {
|
||
this.temp_disable = false;
|
||
}
|
||
|
||
canLog(component) {
|
||
return this.canLogFile(component) || this.canLogConsole(component);
|
||
}
|
||
|
||
canLogFile(component) {
|
||
if (!this.conf.fileOptions.enabled || this.temp_disable) {
|
||
return false;
|
||
}
|
||
if (performance.now() > this.stopTime) {
|
||
return false;
|
||
}
|
||
if (Array.isArray(component) && component.length ) {
|
||
for (const c of component) {
|
||
if (this.conf.fileOptions[c]) {
|
||
return this.conf.fileOptions[c];
|
||
}
|
||
}
|
||
} else {
|
||
return this.conf.fileOptions[component];
|
||
}
|
||
}
|
||
canLogConsole(component) {
|
||
if (!this.conf.consoleOptions.enabled || this.temp_disable) {
|
||
return false;
|
||
}
|
||
if (performance.now() > this.stopTime) {
|
||
return false;
|
||
}
|
||
if (Array.isArray(component) && component.length) {
|
||
for (const c of component) {
|
||
if (this.conf.consoleOptions[c]) {
|
||
return this.conf.consoleOptions[c];
|
||
}
|
||
}
|
||
} else {
|
||
return this.conf.consoleOptions[component] !== undefined ? this.conf.consoleOptions[component] : this.conf.logAll;
|
||
}
|
||
|
||
return this.conf.logAll;
|
||
}
|
||
|
||
// level is unused as of now, but this may change in the future
|
||
// levels: 'info', 'warn', 'error'
|
||
log(level, component, ...message) {
|
||
if (!this.conf) {
|
||
return;
|
||
}
|
||
const error = new Error();
|
||
if (this.conf.logToFile) {
|
||
if (this.canLogFile(component)) {
|
||
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),
|
||
stack: error.stack,
|
||
})
|
||
}
|
||
}
|
||
if (this.conf.logToConsole) {
|
||
if (this.canLogConsole(component)) {
|
||
console.log(...message, error.stack);
|
||
}
|
||
}
|
||
}
|
||
|
||
// 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 --------------------------------------"
|
||
});
|
||
}
|
||
}
|
||
|
||
|
||
}
|
||
|
||
export default Logger;
|