631 lines
18 KiB
TypeScript
631 lines
18 KiB
TypeScript
import { decycle } from 'json-cyclic';
|
||
import Comms from './comms/Comms';
|
||
import BrowserDetect from '../conf/BrowserDetect';
|
||
import { browser } from 'webextension-polyfill-ts';
|
||
|
||
if (process.env.CHANNEL !== 'stable'){
|
||
console.info('Loading Logger');
|
||
}
|
||
|
||
|
||
export const baseLoggingOptions: LoggerConfig = {
|
||
isContentScript: true,
|
||
allowLogging: true,
|
||
useConfFromStorage: true,
|
||
fileOptions: {
|
||
enabled: false
|
||
},
|
||
consoleOptions: {
|
||
"enabled": true,
|
||
"debug": true,
|
||
"init": true,
|
||
"settings": true,
|
||
"keyboard": true,
|
||
"mousemove": false,
|
||
"actionHandler": true,
|
||
"comms": true,
|
||
"playerDetect": true,
|
||
"resizer": true,
|
||
"scaler": true,
|
||
"stretcher": true,
|
||
// "videoRescan": true,
|
||
// "playerRescan": true,
|
||
"arDetect": true,
|
||
"arDetect_verbose": true
|
||
},
|
||
allowBlacklistedOrigins: {
|
||
'periodicPlayerCheck': false,
|
||
'periodicVideoStyleChangeCheck': false,
|
||
'handleMouseMove': false
|
||
}
|
||
};
|
||
|
||
export interface LoggingOptions {
|
||
enabled?: boolean;
|
||
debug?: boolean;
|
||
init?: boolean;
|
||
settings?: boolean;
|
||
keyboard?: boolean;
|
||
mousemove?: boolean;
|
||
actionHandler?: boolean;
|
||
comms?: boolean;
|
||
playerDetect?: boolean;
|
||
resizer?: boolean;
|
||
scaler?: boolean;
|
||
stretcher?: boolean;
|
||
videoRescan?: boolean;
|
||
playerRescan?: boolean;
|
||
arDetect?: boolean;
|
||
arDetect_verbose?: boolean;
|
||
}
|
||
|
||
export interface LoggerBlacklistedOrigins {
|
||
periodicPlayerCheck?: boolean;
|
||
periodicVideoStyleChangeCheck?: boolean;
|
||
handleMouseMove?: boolean;
|
||
}
|
||
|
||
export interface LoggerConfig {
|
||
isContentScript?: boolean;
|
||
isBackgroundScript?: boolean;
|
||
allowLogging?: boolean;
|
||
useConfFromStorage?: boolean;
|
||
fileOptions?: LoggingOptions;
|
||
consoleOptions?: LoggingOptions;
|
||
allowBlacklistedOrigins?: LoggerBlacklistedOrigins;
|
||
}
|
||
|
||
class Logger {
|
||
temp_disable: boolean = false;
|
||
onLogEndCallbacks: any[] = [];
|
||
history: any[] = [];
|
||
globalHistory: any = {};
|
||
isContentScript: boolean = false;
|
||
isBackgroundScript: boolean = true;
|
||
vuexStore: any;
|
||
uwInstance: any;
|
||
conf: any;
|
||
startTime: number;
|
||
stopTime: number;
|
||
|
||
constructor(options?: {vuexStore?: any, uwInstance?: any}) {
|
||
this.vuexStore = options?.vuexStore;
|
||
this.uwInstance = options?.uwInstance;
|
||
}
|
||
|
||
static saveConfig(conf: LoggerConfig) {
|
||
if (process.env.CHANNEL === 'dev') {
|
||
console.info('Saving logger conf:', conf)
|
||
}
|
||
|
||
browser.storage.local.set( {'uwLogger': JSON.stringify(conf)});
|
||
}
|
||
|
||
static syncConfig(callback: (x) => void) {
|
||
browser.storage.onChanged.addListener( (changes, area) => {
|
||
if (changes.uwLogger) {
|
||
const newLoggerConf = JSON.parse(changes.uwLogger.newValue)
|
||
if (process.env.CHANNEL === 'dev') {
|
||
console.info('Logger settings reloaded. New conf:', newLoggerConf);
|
||
}
|
||
callback(newLoggerConf);
|
||
}
|
||
});
|
||
}
|
||
|
||
static async getConfig() {
|
||
let ret;
|
||
|
||
// if (BrowserDetect.firefox) {
|
||
ret = await browser.storage.local.get('uwLogger');
|
||
// } else if (BrowserDetect.anyChromium) {
|
||
// ret = await new Promise( (resolve, reject) => {
|
||
// browser.storage.local.get('uwLogger', (res) => resolve(res));
|
||
// });
|
||
// }
|
||
|
||
if (process.env.CHANNEL === 'dev') {
|
||
try {
|
||
console.info("[Logger::getSaved] Got settings:", JSON.parse(ret.uwLogger));
|
||
} catch (e) {
|
||
console.info("[Logger::getSaved] No settings.")
|
||
}
|
||
}
|
||
|
||
try {
|
||
return JSON.parse(ret.uwLogger);
|
||
} catch(e) {
|
||
return {consoleOptions: {}, fileOptions: {}};
|
||
}
|
||
}
|
||
|
||
async init(conf: LoggerConfig) {
|
||
// 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
|
||
if (conf.isContentScript) {
|
||
this.isContentScript = true;
|
||
this.isBackgroundScript = false;
|
||
} else if (conf.isBackgroundScript) {
|
||
this.isContentScript = false;
|
||
this.isBackgroundScript = true;
|
||
}
|
||
|
||
if (conf && process.env.CHANNEL === 'dev' && !conf.useConfFromStorage) {
|
||
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.startTime = performance.now();
|
||
this.temp_disable = false;
|
||
this.stopTime = this.conf.timeout ? performance.now() + (this.conf.timeout * 1000) : undefined;
|
||
|
||
browser.storage.onChanged.addListener( (changes, area) => {
|
||
if (process.env.CHANNEL === 'dev') {
|
||
if (!changes.uwLogger) {
|
||
// console.info('[Logger::<storage/on change> No new logger settings!');
|
||
}
|
||
if (changes['uwLogger'] && changes['uwLogger'].newValue) {
|
||
// console.log("[Logger::<storage/on change>] Logger have been changed outside of here. Updating active settings. Changes:", changes, "storage area:", area);
|
||
// console.info("[Logger::<storage/on change>] new logger settings object (parsed):", JSON.parse(changes.uwLogger.newValue));
|
||
}
|
||
}
|
||
if (!changes['uwLogger']) {
|
||
return;
|
||
}
|
||
if (changes['uwLogger']?.newValue) {
|
||
const newConf = JSON.parse(changes.uwLogger.newValue);
|
||
if (this.isContentScript && this.conf.allowLogging && !newConf.allowLogging) {
|
||
this.saveToVuex();
|
||
}
|
||
this.conf = newConf;
|
||
}
|
||
});
|
||
}
|
||
|
||
setVuexStore(store) {
|
||
this.vuexStore = store;
|
||
}
|
||
|
||
clear() {
|
||
this.history = [];
|
||
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:
|
||
Logger.saveConfig(conf);
|
||
}
|
||
|
||
// async getSaved() {
|
||
// return Logger.getSaved();
|
||
// }
|
||
|
||
|
||
// 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.stringify(this.history),
|
||
}
|
||
}
|
||
|
||
pause() {
|
||
this.temp_disable = true;
|
||
}
|
||
resume() {
|
||
this.temp_disable = false;
|
||
}
|
||
|
||
onLogEnd(callback) {
|
||
this.onLogEndCallbacks.push(callback);
|
||
}
|
||
|
||
// this should be used mostly in background page instance of logger, btw
|
||
//
|
||
addToGlobalHistory(key, log) {
|
||
this.globalHistory[key] = log;
|
||
this.log('info', 'debug', 'Added log for', key, 'to global history. Current history:', this.globalHistory);
|
||
}
|
||
|
||
finish() {
|
||
if (!this.isBackgroundScript) {
|
||
this.conf.allowLogging = false;
|
||
// const logJson = JSON.stringify(decycle(this.history));
|
||
// this.log('force', 'debugr', 'Calling all log end callbacks. There\'s this many of them:', 1);
|
||
// for(const f of this.onLogEndCallbacks) {
|
||
// f(logJson);
|
||
// }
|
||
// } else {
|
||
// this.exportLogToFile();
|
||
}
|
||
this.saveViaBgScript();
|
||
}
|
||
|
||
parseStack() {
|
||
const trace = (new Error()).stack;
|
||
|
||
const stackInfo: any = {};
|
||
// we turn our stack into array and remove the "file::line" part of the trace,
|
||
// since that is useless because minification/webpack
|
||
stackInfo['stack'] = {trace: trace.split('\n').map(a => a.split('@')[0])};
|
||
|
||
// here's possible sources that led to this log entry
|
||
stackInfo['periodicPlayerCheck'] = false;
|
||
stackInfo['periodicVideoStyleChangeCheck'] = false;
|
||
stackInfo['aard'] = false;
|
||
stackInfo['keyboard'] = false;
|
||
stackInfo['popup'] = false;
|
||
stackInfo['mousemove'] = false;
|
||
stackInfo['exitLogs'] = false;
|
||
|
||
// here we check which source triggered the action. There can be more
|
||
// than one source, too, so we don't break when we find the first one
|
||
for (const line of stackInfo.stack.trace) {
|
||
if (line === 'doPeriodicPlayerElementChangeCheck') {
|
||
stackInfo['periodicPlayerCheck'] = true;
|
||
} else if (line === 'doPeriodicFallbackChangeDetectionCheck') {
|
||
stackInfo['periodicVideoStyleChangeCheck'] = true;
|
||
} else if (line === 'frameCheck') {
|
||
stackInfo['aard'] = true;
|
||
} else if (line === 'execAction') {
|
||
stackInfo['keyboard'] = true;
|
||
} else if (line === 'processReceivedMessage') {
|
||
stackInfo['popup'] = true;
|
||
} else if (line === 'handleMouseMove') {
|
||
stackInfo['mousemove'] = true;
|
||
}
|
||
}
|
||
|
||
// exitLog overrides any other exclusions, so we look for it separately.
|
||
// we also remove some of the unnecessary messages to reduce log file size
|
||
for(let i = 0; i < stackInfo.stack.trace.length; i++) {
|
||
if (stackInfo.stack.trace[i] === 'finish') {
|
||
stackInfo['exitLogs'] = true;
|
||
break;
|
||
}
|
||
|
||
// if we hit one of these, we remove the rest of the array and call it a
|
||
// day. Chances are there's nothing of value past this point.
|
||
if (stackInfo.stack.trace[i].indexOf('promise callback') !== -1
|
||
|| stackInfo.stack.trace[i].indexOf('asyncGeneratorStep') !== -1
|
||
|| stackInfo.stack.trace[i].indexOf('_asyncToGenerator') !== -1
|
||
|| stackInfo.stack.trace[i].startsWith('_next')) {
|
||
stackInfo.stack.trace.splice(i);
|
||
break;
|
||
}
|
||
}
|
||
|
||
return stackInfo;
|
||
}
|
||
|
||
// test for blacklisted origin
|
||
isBlacklistedOrigin(stackInfo) {
|
||
if (stackInfo.periodicPlayerCheck) {
|
||
return !this.conf.allowBlacklistedOrigins?.periodicPlayerCheck;
|
||
}
|
||
if (stackInfo.periodicVideoStyleChangeCheck) {
|
||
return !this.conf.allowBlacklistedOrigins?.periodicVideoStyleChangeCheck;
|
||
}
|
||
if (stackInfo.mousemove) {
|
||
return !this.conf.allowBlacklistedOrigins?.handleMouseMove;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
isTimeUp() {
|
||
// we don't do timeouts on background script
|
||
if (this.isBackgroundScript) {
|
||
return false;
|
||
}
|
||
if (this.stopTime && performance.now() > this.stopTime) {
|
||
if (this.conf.allowLogging) {
|
||
this.log('force', 'debug', '-----[ alloted time has run out. logging will stop ]-----');
|
||
this.finish();
|
||
}
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
isLoggingAllowed() {
|
||
return this.conf.allowLogging;
|
||
}
|
||
|
||
isLoggingToFile() {
|
||
return this.conf.allowLogging && this.conf.fileOptions?.enabled;
|
||
}
|
||
|
||
// NOTE: THIS FUNCTION IS NEVER USED INTERNALLY!
|
||
canLog(component) {
|
||
const stackInfo = this.parseStack();
|
||
|
||
if (!this.conf.allowLogging && !stackInfo.exitLogs) {
|
||
return false;
|
||
}
|
||
|
||
if (this.isBlacklistedOrigin(stackInfo)) {
|
||
return false;
|
||
}
|
||
|
||
// if either of these two is true, we allow logging to happen (forbidden origins were checked above)
|
||
return (this.canLogFile(component) || this.canLogConsole(component) || stackInfo.exitLogs);
|
||
}
|
||
|
||
canLogFile(component) {
|
||
if (!this.conf.fileOptions.enabled || this.temp_disable) {
|
||
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, stackInfo?) {
|
||
if (!this.conf.consoleOptions?.enabled || this.temp_disable) {
|
||
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;
|
||
}
|
||
|
||
logToFile(message, stackInfo) {
|
||
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: message,
|
||
stack: stackInfo,
|
||
});
|
||
}
|
||
|
||
logToConsole(level, message, stackInfo) {
|
||
try {
|
||
switch (level) {
|
||
case 'error':
|
||
console.error(...message, {stack: stackInfo});
|
||
break;
|
||
case 'warn':
|
||
console.warn(...message, {stack: stackInfo});
|
||
break;
|
||
case 'info':
|
||
console.info(...message, {stack: stackInfo});
|
||
break;
|
||
default:
|
||
console.log(...message, {stack: stackInfo});
|
||
}
|
||
} catch (e) {
|
||
console.error("Message too big to log. Error:", e, "stackinfo:", stackInfo);
|
||
}
|
||
}
|
||
|
||
// level is unused as of now, but this may change in the future
|
||
// levels: 'info', 'warn', 'error'.
|
||
// if level is `true` (bool), logging happens regardless of any other
|
||
// settings
|
||
log(level, component, ...message) {
|
||
const stackInfo = this.parseStack();
|
||
|
||
if (!this.conf || (!this.conf.allowLogging && !stackInfo.exitLogs)) {
|
||
return;
|
||
}
|
||
|
||
// skip all checks if we force log
|
||
if (level === 'force') {
|
||
if (this.conf.fileOptions?.enabled) {
|
||
this.logToFile(message, stackInfo);
|
||
}
|
||
if (this.conf.consoleOptions?.enabled) {
|
||
this.logToConsole(level, message, stackInfo);
|
||
}
|
||
return; // don't check further — recursion-land ahead!
|
||
}
|
||
|
||
if (this.isTimeUp() && !stackInfo.exitLogs) {
|
||
return;
|
||
}
|
||
|
||
// don't log stuff from blacklisted origin (unless logger conf says otherwise)
|
||
if (this.isBlacklistedOrigin(stackInfo)) {
|
||
return;
|
||
}
|
||
|
||
if (this.conf.fileOptions?.enabled) {
|
||
if (this.canLogFile(component) || stackInfo.exitLogs) {
|
||
this.logToFile(message, stackInfo);
|
||
}
|
||
}
|
||
if (this.conf.consoleOptions?.enabled) {
|
||
if (this.canLogConsole(component) || stackInfo.exitLogs) {
|
||
this.logToConsole(level, message, stackInfo);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
// 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 --------------------------------------"
|
||
});
|
||
}
|
||
}
|
||
|
||
appendLog(logs) {
|
||
this.history = this.history.concat(logs);
|
||
}
|
||
|
||
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);
|
||
}
|
||
}
|
||
|
||
saveViaBgScript() {
|
||
console.info('[info] will attempt to save. Issuing "show-logger"');
|
||
if (this.isBackgroundScript) {
|
||
console.info('[info] Background script cannot display logger UI.');
|
||
return;
|
||
}
|
||
// if (!this.conf?.fileOptions?.enabled) {
|
||
// console.info('[info] Logging to file is disabled. Logger won\'t get shown.');
|
||
// return;
|
||
// }
|
||
|
||
Comms.sendMessage({cmd: 'show-logger', forwardToSameFramePort: true, port: 'content-ui-port'});
|
||
|
||
let exportObject;
|
||
try {
|
||
exportObject = {
|
||
pageLogs: decycle(this.history),
|
||
backgroundLogs: decycle(this.globalHistory),
|
||
loggerFileOptions: this.conf.fileOptions,
|
||
}
|
||
} catch (e) {
|
||
console.error("[fail] error parsing logs!", e)
|
||
return;
|
||
}
|
||
|
||
try {
|
||
Comms.sendMessage({cmd: 'emit-logs', payload: JSON.stringify(exportObject), forwardToSameFramePort: true, port: 'content-ui-port'})
|
||
} catch (e) {
|
||
console.log("failed to send message")
|
||
}
|
||
}
|
||
|
||
saveToVuex() {
|
||
console.info('[info] will attempt to save to vuex store.');
|
||
if (!this.conf?.fileOptions?.enabled || this.isBackgroundScript) {
|
||
console.info('[info] Logging to file is either disabled or we\'re not on the content script. Not saving.');
|
||
return;
|
||
}
|
||
|
||
if (!this.vuexStore) {
|
||
console.error("[info] No vue store.");
|
||
if (!this.uwInstance) {
|
||
console.error('[info] no vue instance either. Not logging.');
|
||
return;
|
||
}
|
||
console.info("[info] Initializing vue and vuex instance ...");
|
||
this.uwInstance.initVue();
|
||
}
|
||
|
||
console.info('[info] vuex store present. Parsing logs.');
|
||
|
||
let exportObject;
|
||
try {
|
||
exportObject = {
|
||
pageLogs: decycle(this.history),
|
||
backgroundLogs: decycle(this.globalHistory),
|
||
loggerFileOptions: this.conf.fileOptions,
|
||
}
|
||
} catch (e) {
|
||
console.error("[fail] error parsing logs!", e)
|
||
return;
|
||
}
|
||
|
||
console.info('[info] Logs were parsed successfuly. Putting stuff to vuex ...');
|
||
try {
|
||
this.vuexStore.dispatch('uw-set-log', exportObject);
|
||
} catch (e) {
|
||
console.log("[fail] error saving to vuex", e);
|
||
return;
|
||
}
|
||
|
||
console.info('[info] Export object saved to vuex store.')
|
||
}
|
||
}
|
||
|
||
if (process.env.CHANNEL !== 'stable'){
|
||
console.info('Logger loaded');
|
||
}
|
||
|
||
export default Logger;
|