2019-09-03 21:30:18 +02:00
import Debug from '../conf/Debug' ;
2019-09-03 00:48:18 +02:00
import currentBrowser from '../conf/BrowserDetect' ;
2019-07-16 20:59:12 +02:00
class Logger {
constructor ( conf ) {
}
2020-01-16 01:00:47 +01:00
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 ) {
2020-01-27 23:20:38 +01:00
return { consoleOptions : { } , fileOptions : { } } ;
2020-01-16 01:00:47 +01:00
}
}
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 = { } ;
}
2020-01-27 23:20:38 +01:00
if ( this . conf . fileOptions === undefined ) {
2020-01-27 22:32:38 +01:00
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 ;
2019-09-03 00:48:18 +02:00
const ths = this ;
const br = currentBrowser . firefox ? browser : chrome ;
2020-01-27 23:20:38 +01:00
2019-09-03 00:48:18 +02:00
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 ) ;
}
} ) ;
2020-01-16 01:00:47 +01:00
2019-09-03 00:48:18 +02:00
}
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 ) {
2019-09-03 00:48:18 +02:00
this . conf = conf ; // effective immediately
// also persist settings:
2020-01-16 01:00:47 +01:00
Logger . saveConfig ( conf ) ;
2019-07-16 20:59:12 +02:00
}
2019-09-03 00:48:18 +02:00
async getSaved ( ) {
2020-01-16 01:00:47 +01:00
return Logger . getSaved ( ) ;
2019-09-03 00:48:18 +02:00
}
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
}
}
2019-07-18 21:25:58 +02:00
pause ( ) {
this . temp _disable = true ;
}
resume ( ) {
this . temp _disable = false ;
}
2020-01-28 01:27:30 +01:00
parseStack ( ) {
const trace = ( new Error ( ) ) . stack ;
const stackInfo = { } ;
// 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 ;
// here we check which source triggered the action. We know that only one of these
// functions will appear in the trace at most once (and if more than one of these
// appears — e.g. frameCheck triggered by user toggling autodetection in popup —
// the most recent one will be the correct one 99% of the time)
for ( const line of stackInfo . stack . trace ) {
if ( line . startsWith ( 'doPeriodicPlayerElementChangeCheck' ) ) {
stackInfo [ 'periodicPlayerCheck' ] = true ;
break ;
} else if ( line . startsWith ( 'doPeriodicFallbackChangeDetectionCheck' ) ) {
stackInfo [ 'periodicVideoStyleChangeCheck' ] = true ;
break ;
} else if ( line . startsWith ( 'frameCheck' ) ) {
stackInfo [ 'aard' ] = true ;
break ;
} else if ( line . startsWith ( 'execAction' ) ) {
stackInfo [ 'keyboard' ] = true ;
break ;
} else if ( line . startsWith ( 'processReceivedMessage' ) ) {
stackInfo [ 'popup' ] = true ;
break ;
}
}
return stackInfo ;
}
canLog ( component , stackInfo ) {
if ( ! this . conf . allowLogging ) {
return false ;
}
// if we call this function from outside of logger,
// stackInfo may not be provided. Here's a fallback.
if ( stackInfo === undefined ) {
stackInfo = this . parseStack ( ) ;
}
// check if log belongs to a blacklisted origin. If yes, then only allow content to be logged if the
// origin is explicitly whitelisted in the conf object
if ( stackInfo ) {
if ( stackInfo . periodicPlayerCheck ) {
return this . conf . allowBlacklistedOrigins && this . conf . allowBlacklistedOrigins . periodicPlayerCheck ;
}
if ( stackInfo . periodicVideoStyleChangeCheck ) {
return this . conf . allowBlacklistedOrigins && this . conf . allowBlacklistedOrigins . periodicVideoStyleChangeCheck ;
}
}
// if either of these two is true, we allow logging to happen (forbidden origins were checked above)
return ( this . canLogFile ( component ) || this . canLogConsole ( component ) ) ;
2019-07-18 21:25:58 +02:00
}
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 ) {
2020-01-16 01:00:47 +01:00
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-07-18 21:25:58 +02:00
}
}
2019-11-04 22:14:41 +01:00
} else {
return this . conf . fileOptions [ component ] ;
}
2019-07-18 21:25:58 +02:00
}
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 ) {
2020-01-16 01:00:47 +01:00
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-07-16 22:46:16 +02:00
}
}
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 22:46:16 +02:00
}
2019-07-16 20:59:12 +02:00
// level is unused as of now, but this may change in the future
2019-07-16 22:46:16 +02:00
// levels: 'info', 'warn', 'error'
2019-07-16 20:59:12 +02:00
log ( level , component , ... message ) {
2019-07-16 22:46:16 +02:00
if ( ! this . conf ) {
return ;
}
2020-01-28 01:27:30 +01:00
const stackInfo = this . parseStack ( ) ;
2020-01-27 23:20:38 +01:00
if ( this . conf . fileOptions . enabled ) {
2020-01-28 01:27:30 +01:00
if ( this . canLogFile ( component , stackInfo ) ) {
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-28 01:27:30 +01:00
stack : stackInfo ,
2019-07-16 20:59:12 +02:00
} )
}
}
2020-01-27 23:20:38 +01:00
if ( this . conf . consoleOptions . enabled ) {
2020-01-28 01:27:30 +01:00
if ( this . canLogConsole ( component , stackInfo ) ) {
console . log ( ... message , { stack : stackInfo } ) ;
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 ;