whistle
Version:
HTTP, HTTP2, HTTPS, Websocket debugging proxy
443 lines (390 loc) • 11.9 kB
JavaScript
;(function() {
if (typeof window === 'undefined' || typeof Image === 'undefined') {
return;
}
if (window._whistleConsole) {
return;
}
var console = window.console = window.console || {};
var wConsole = window._whistleConsole = {};
var JSON = window.JSON || patchJSON();
function patchJSON() {
var JSON = {};
var rx_escapable = /[\\\"\u0000-\u001f\u007f-\u009f\u00ad\u0600-\u0604\u070f\u17b4\u17b5\u200c-\u200f\u2028-\u202f\u2060-\u206f\ufeff\ufff0-\uffff]/g;
function f(n) {
return n < 10
? '0' + n
: n;
}
function this_value() {
return this.valueOf();
}
if (typeof Date.prototype.toJSON !== 'function') {
Date.prototype.toJSON = function () {
return isFinite(this.valueOf())
? this.getUTCFullYear() + '-' +
f(this.getUTCMonth() + 1) + '-' +
f(this.getUTCDate()) + 'T' +
f(this.getUTCHours()) + ':' +
f(this.getUTCMinutes()) + ':' +
f(this.getUTCSeconds()) + 'Z'
: null;
};
Boolean.prototype.toJSON = this_value;
Number.prototype.toJSON = this_value;
String.prototype.toJSON = this_value;
}
var gap,
indent,
meta,
rep;
function quote(string) {
rx_escapable.lastIndex = 0;
return rx_escapable.test(string)
? '"' + string.replace(rx_escapable, function (a) {
var c = meta[a];
return typeof c === 'string'
? c
: '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4);
}) + '"'
: '"' + string + '"';
}
function str(key, holder) {
var i,
k,
v,
length,
mind = gap,
partial,
value = holder[key];
if (value && typeof value === 'object' &&
typeof value.toJSON === 'function') {
value = value.toJSON(key);
}
if (typeof rep === 'function') {
value = rep.call(holder, key, value);
}
switch (typeof value) {
case 'string':
return quote(value);
case 'number':
return isFinite(value)
? String(value)
: 'null';
case 'boolean':
case 'null':
return String(value);
case 'object':
if (!value) {
return 'null';
}
gap += indent;
partial = [];
if (Object.prototype.toString.apply(value) === '[object Array]') {
length = value.length;
for (i = 0; i < length; i += 1) {
partial[i] = str(i, value) || 'null';
}
v = partial.length === 0
? '[]'
: gap
? '[\n' + gap + partial.join(',\n' + gap) + '\n' + mind + ']'
: '[' + partial.join(',') + ']';
gap = mind;
return v;
}
if (rep && typeof rep === 'object') {
length = rep.length;
for (i = 0; i < length; i += 1) {
if (typeof rep[i] === 'string') {
k = rep[i];
v = str(k, value);
if (v) {
partial.push(quote(k) + (
gap
? ': '
: ':'
) + v);
}
}
}
} else {
for (k in value) {
if (Object.prototype.hasOwnProperty.call(value, k)) {
v = str(k, value);
if (v) {
partial.push(quote(k) + (
gap
? ': '
: ':'
) + v);
}
}
}
}
v = partial.length === 0
? '{}'
: gap
? '{\n' + gap + partial.join(',\n' + gap) + '\n' + mind + '}'
: '{' + partial.join(',') + '}';
gap = mind;
return v;
}
}
if (typeof JSON.stringify !== 'function') {
meta = {
'\b': '\\b',
'\t': '\\t',
'\n': '\\n',
'\f': '\\f',
'\r': '\\r',
'"': '\\"',
'\\': '\\\\'
};
JSON.stringify = function (value, replacer, space) {
var i;
gap = '';
indent = '';
if (typeof space === 'number') {
for (i = 0; i < space; i += 1) {
indent += ' ';
}
} else if (typeof space === 'string') {
indent = space;
}
rep = replacer;
if (replacer && typeof replacer !== 'function' &&
(typeof replacer !== 'object' ||
typeof replacer.length !== 'number')) {
throw new Error('JSON.stringify');
}
return str('', {'': value});
};
}
return JSON;
}
function stringify(obj) {
if (typeof obj === 'function') {
return obj.toString();
}
if (obj instanceof Error) {
var stack = obj.stack;
if (stack && typeof stack === 'string') {
if (obj.message && stack.indexOf(obj.message) === -1) {
return 'Error: ' + obj.message + '\n' + stack;
}
}
return 'Error: ' + obj.message;
}
return obj === undefined ? 'undefined' : obj;
}
var prefixPath;
function getPathPrefix() {
if (prefixPath) {
return prefixPath;
}
prefixPath = window.__WHISTLE_PATH_PREFIX__;
if (/^\/[\w./-]+$/.test(prefixPath) && prefixPath.length <= 128) {
var len = prefixPath.length - 1;
if (prefixPath[len] === '/') {
prefixPath = prefixPath.substring(0, len);
}
} else {
prefixPath = '';
}
return prefixPath;
}
var MAX_SIZE = 1024 * 512;
var MAX_VAL_SIZE = 1024 * 64;
var LOG_ID = '$LOG_ID';
var cacheList = [];
var origin = (location.origin || (location.protocol + '//' + location.host)) + '/';
function setLog() {
if (!cacheList.length) {
return;
}
var list = cacheList.splice(0, 10);
var xhr = new XMLHttpRequest();
var url = '$BASE_URL' + getPathPrefix() + '$LOG_CGI';
xhr.open('POST', url);
xhr.setRequestHeader('Content-Type', url.indexOf(origin) ? 'text/plain' : 'application/json');
xhr.onload = function() {};
xhr.send(JSON.stringify({ list: list }));
}
function addLog(level, text) {
cacheList.push([new Date().getTime(), level, LOG_ID, text].join('\r'));
if (typeof window.onWhistleLogSend === 'function') {
window.onWhistleLogSend(level, text);
}
setTimeout(setLog, 60);
}
function getPageInfo() {
return '\r\nPage URL: ' + location.href + '\r\nUser Agent: ' + navigator.userAgent;
}
function getErrorStack(error, message) {
var stack = (error.stack || error.message) + '';
message = message || error.message;
if (typeof message === 'string') {
var msg = message.substring(message.indexOf(':') + 1);
if (stack.indexOf(msg) === -1) {
stack = message + '\n' + stack;
}
}
return stack + getPageInfo();
}
function arrayIndexOf(arr, value) {
if (arr.indexOf) {
return arr.indexOf(value);
}
for (var i = 0, len = arr.length; i < len; i++) {
if (arr[i] === value) {
return i;
}
}
return -1;
}
function fomatText(str, size) {
var len = str.length;
if (len <= size + 9) {
return str;
}
return str.substring(0, size) + '...(' + (len - size) + ')';
}
function stringifyObj(obj) {
if (typeof obj === 'string') {
return obj;
}
var str;
try {
str = JSON.stringify(obj);
if (str.length <= MAX_SIZE) {
return str;
}
} catch(e) {}
try {
var keyList = [];
var valList = [];
str = JSON.stringify(obj, function(key, value) {
if (!value) {
return value;
}
var type = typeof value;
if (type === 'string' && value.length > MAX_VAL_SIZE) {
return fomatText(value, MAX_VAL_SIZE);
}
if (type === 'object') {
var index = arrayIndexOf(valList, value);
valList.push(value);
keyList.push(key);
if (index !== -1) {
return '[Circular ' + keyList[index] + ']';
}
}
return value;
});
return fomatText(str, MAX_SIZE);
} catch(e) {}
}
var levels = ['fatal', 'error', 'warn', 'info', 'debug', 'log'];
var noop = function() {};
var slice = Array.prototype.slice;
for (var i = 0, len = levels.length; i < len; i++) {
(function(level) {
var fn = console[level] || noop;
var pending;
var wFn = wConsole[level] = function() {
var result = slice.call(arguments);
if (!result.length) {
result = [undefined];
}
if (typeof window.onBeforeWhistleLogSend === 'function') {
pending = true;
try {
window.onBeforeWhistleLogSend(result, level);
} catch(e) {
result.push('onBeforeWhistleLogSend' + stringify(e));
} finally {
pending = false;
}
}
var len = result.length;
if (!len) {
return;
}
for (var i = 0; i < len; i++) {
result[i] = stringify(result[i]);
}
result = stringifyObj(result);
result && addLog(level, result);
};
if ($INTERCEPT_CONSOLE) {
console[level] = function() {
if (pending) {
return;
}
pending = true;
var weinreFn = console['_weinre_' + level];
if (typeof weinreFn === 'function') {
try {
weinreFn.apply(this, arguments);
} catch (e) {}
}
wFn.apply(null, arguments);
try {
fn.apply(this, arguments);
} catch(e) {
fn(arguments.length < 2 ? arguments[0] : slice.apply(arguments));
} finally {
pending = false;
}
};
}
})(levels[i]);
}
/*eslint no-console: "off"*/
var onerror = function(message, filename, lineno, colno, error) {
if (error) {
wConsole.error(getErrorStack(error, message));
} else {
wConsole.error('Error: ' + message + '(' + filename
+ ':' + lineno + ':' + (colno || 0) + ')' + getPageInfo());
}
};
var isWeinreConsole = function(curConsole) {
if (curConsole === console || typeof curConsole.log !== 'function') {
return;
}
return curConsole.log.toString().indexOf('this._generic(MessageLevel.Log,') !== -1;
};
var attachOnError = function() {
if (window.onerror !== onerror) {
window.onerror = onerror;
}
var curConsole = window.console;
if (!curConsole) {
window.console = console;
} else if ($INTERCEPT_CONSOLE && isWeinreConsole(curConsole)) {
for (var i = 0, len = levels.length; i < len; i++) {
var level = levels[i];
var fn = console[level];
var curFn = curConsole[level];
if (fn !== curFn) {
console['_weinre_' + level] = curFn;
curConsole[level] = fn;
}
}
}
setTimeout(attachOnError, 600);
};
attachOnError();
if (typeof window.addEventListener === 'function') {
window.addEventListener('unhandledrejection', function(e) {
var reason = 'UnhandledRejection';
if (e) {
e = stringifyObj(e.reason || e) || String(e);
reason += (/^[\w.-]*:/.test(e) ? ' ' : ': ') + e;
}
wConsole.error(reason);
});
}
})();