// ==UserScript==
// @name 浏览器控制台防检测
// @namespace https://gf.qytechs.cn/users/667968-pyudng
// @version 0.4
// @description 根据 https://github.com/AEPKILL/devtools-detector 的检测方法进行了一个逆向反检测...需要在哪些网站上运行,自己添加到脚本编辑器-设置-用户包括里面去
// @author PY-DNG
// @license MIT
// @match http*://AddYour.OwnMatch/
// @match http*://blog.aepkill.com/demos/devtools-detector/
// @require https://update.gf.qytechs.cn/scripts/456034/1507114/Basic%20Functions%20%28For%20userscripts%29.js
// @icon 
// @grant GM_registerMenuCommand
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_addElement
// @run-at document-start
// ==/UserScript==
/* eslint-disable no-multi-spaces */
/* eslint-disable no-return-assign */
/* global LogLevel DoLog Err $ $All $CrE $AEL $$CrE addStyle detectDom destroyEvent copyProp copyProps parseArgs escJsStr replaceText getUrlArgv dl_browser dl_GM AsyncManager queueTask testChecker registerChecker loadFuncs */
(function __MAIN__() {
'use strict';
const CONST = {
TextAllLang: {
DEFAULT: 'zh-CN',
'zh-CN': {}
}
};
// Init language
const i18n = Object.keys(CONST.TextAllLang).includes(navigator.language) ? navigator.language : CONST.TextAllLang.DEFAULT;
CONST.Text = CONST.TextAllLang[i18n];
// As a common dependency for almost all funcs, load it synchrously to execute funcs as soon as poosible
// If we write utils as a normal func which executes in loadFuncs() and serves as a func module through require('utils'),
// func that depends on utils must wait until utils load asynchrously, which will takes a lot of time.
// Execute as soon as possible is essential to ensure we have time advantage agains devtools detectors.
const utils = (function() {
const win = typeof unsafeWindow === 'undefined' ? window : unsafeWindow;
const SavedValues = {
_targets: ['Object', 'Function', 'Symbol', 'Reflect', 'performance', 'setTimeout'],
_thiskey: Symbol('SavedValues.this-key'),
original: {}, // original property descriptors
bound: {} // property descriptors with "this" bound for whose value is function
};
for (const target_name of SavedValues._targets) {
const source = win[target_name];
if (['object', 'function'].includes(typeof source) && source !== null) {
// Save original
const obj_original = SavedValues.original[target_name] = { [SavedValues._thiskey]: source };
const obj_bound = SavedValues.bound[target_name] = {}
for (const prop of Reflect.ownKeys(source)) {
const desc_original = Object.getOwnPropertyDescriptor(source, prop);
Object.defineProperty(obj_original, prop, desc_original);
const desc_bound = { ...desc_original };
if (typeof desc_bound.value === 'function') {
desc_bound.value = desc_bound.value.bind(source);
}
Object.defineProperty(obj_bound, prop, desc_bound);
}
} else {
SavedValues.original[target_name] = source;
SavedValues.bound[target_name] = source;
}
}
function hook(obj, prop, func) {
const ori_func = obj[prop];
const hook_func = obj[prop] = function() {
return func.call(this, ori_func, ...arguments);
};
SavedValues.bound.Object.defineProperty(hook_func, 'name', {
get() { return ori_func.name; },
set() { return true; },
configurable: true,
enumerable: true
});
hook_func.toString = function toString() { return ori_func.toString(); }
return {
hook_func, ori_func, unhook
};
function unhook() {
obj[prop] = ori_func;
}
}
return { window: win, SavedValues, hook }
}) ();
const hasOwnProperty = (that, prop) => Object.prototype.hasOwnProperty.call(that, prop);
const FunctionNotLoaded = Symbol('Function-Not-Loaded');
const funcLoaded = id => hasOwnProperty(return_values, id);
const require = id => funcLoaded(id) ? return_values[id] : FunctionNotLoaded;
const return_values = loadFuncs([{
id: 'log-toString-trap',
desc: '处理各种通过自定义toString/字符串getter,再log到console的检测方法',
func: function() {
const console = utils.window.console;
const toStringModified = obj => Function.prototype.toString.call(obj.toString) !== 'function toString() { [native code] }';
utils.hook(console, 'log', function(log, obj) {
if (obj instanceof Date && toStringModified(obj)) {
DoLog(LogLevel.Success, ['拦截了一次 date-toString-trap']);
return log.call(this);
}
if (obj instanceof Date && toStringModified(obj)) {
DoLog(LogLevel.Success, ['拦截了一次 date-toString-trap']);
return log.call(this);
}
if (obj instanceof HTMLElement && utils.SavedValues.original.Object.hasOwnProperty.call(obj, 'id')) {
DoLog(LogLevel.Success, ['拦截了一次 element-id-trap']);
return log.call(this);
}
if (obj instanceof Function && toStringModified(obj)) {
DoLog(LogLevel.Success, ['拦截了一次 function-toString-trap']);
return log.call(this);
}
if (obj instanceof RegExp && toStringModified(obj)) {
DoLog(LogLevel.Success, ['拦截了一次 regexp-toString-trap']);
return log.call(this);
}
const args = [...arguments];
args.shift();
return log.apply(this, args);
});
utils.hook(console, 'table', function(table, obj) {
for (const prop of utils.SavedValues.bound.Reflect.ownKeys(obj)) {
const val = obj[prop];
if (val instanceof RegExp && toStringModified(val)) {
DoLog(LogLevel.Success, ['拦截了一次 dep-reg-trap']);
return table.call(this);
}
}
const args = [...arguments];
args.shift();
return table.apply(this, args);
});
}
}, {
id: 'time-sync',
desc: '处理各种利用时间检测的方法;原理是让获取时间的函数在一次事件循环里返回同一个值',
func: function() {
return {
getTime: {
performance: hookTimeFunc(utils.window.performance, 'now'),
date: hookTimeFunc(utils.window.Date, 'now'),
dateInstance: hookTimeFunc(utils.window.Date, 'getTime', true)
}
};
function hookTimeFunc(obj, funcname, construct=false) {
// 存储当前事件循环里,performance.time返回的值
// 确切地说,是从一次work调用到下一次work调用之间,返回相同的值
let time;
// hook performance.now,始终返回time的值
const getTime = construct ?
(function() {
const func = obj.prototype[funcname];
return function() {
return func.call(new obj(), ...arguments);
};
}) () : obj[funcname].bind(obj);
Object.defineProperty(construct ? obj.prototype : obj, funcname, {
get() { return function() { return time; } },
set() { return true; },
});
// work函数,每次事件循环执行一次,清空
function work() {
const setTimeout = utils.SavedValues.original.setTimeout[utils.SavedValues._thiskey].bind(utils.window);
setTimeout(work);
time = getTime();
}
work();
return getTime;
}
}
}, {
name: 'console-no-clear',
desc: '防止调用console.clear()循环清除控制台',
func: function() {
const console = utils.window.console;
const clear = console._clear = console.clear;
let last_clear = getTime();
const hook = utils.hook(console, 'clear', function() {
const time = getTime();
const time_past = time - last_clear;
if (time_past < 750) {
// prevent clearing
} else {
DoLog(time_past);
clear();
}
last_clear = time;
});
console.log(`[${GM_info.script.name}] 已清除console.clear以防止控制台被清除。如果需要清除控制台,请使用 %cconsole._clear()%c 或者点击控制台清空按钮`, 'color: orange;', '');
function getTime() {
return funcLoaded('time-sync') ? require('time-sync').getTime.performance() : performance.now();
}
return { clear }
}
}, {
name: 'console-filter',
desc: '提供选项供用户拦截控制台日志',
func: function() {
let filter_enabled = isEnabled();
GM_registerMenuCommand('禁止网页输出日志', function() {
filter_enabled = !filter_enabled;
if (filter_enabled) {
enable();
console._log(`%c已禁止网页输出日志到控制台\n%c如需调用console.xxx输出,请在xxx前加一个下划线"_"再调用,如 console._log('Hello, world');`, 'color: green;', 'color: royalblue;');
} else {
disable();
console._log('%c已允许网页输出日志到控制台', 'color: green;');
}
});
const console = utils.window.console;
Object.keys(console).forEach(prop => {
if (['clear', '_clear'].includes(prop)) { return; }
console['_' + prop] = console[prop];
utils.hook(console, prop, function(func, ...args) {
if (!filter_enabled) {
return func.apply(this, args);
}
});
});
return {
get enabled() { return filter_enabled; },
set enabled(val) { val ? enable() : disable(); }
};
function getEnabledHostList() {
const enabled_list = GM_getValue('filtered-hosts', []);
return enabled_list;
}
function saveEnabledHostList(enabled_list) {
GM_setValue('filtered-hosts', enabled_list);
}
function isEnabled() {
const host = location.host;
const enabled_list = getEnabledHostList();
return enabled_list.includes(host);
}
function enable() {
const host = location.host;
const enabled_list = getEnabledHostList();
!enabled_list.includes(host) && enabled_list.push(host);
saveEnabledHostList(enabled_list);
}
function disable() {
const host = location.host;
const enabled_list = getEnabledHostList();
enabled_list.includes(host) && enabled_list.splice(enabled_list.indexOf(host), 1);
saveEnabledHostList(enabled_list);
}
}
}, {
name: 'user-code',
desc: '在文档开始加载时运行用户提供的自定义代码',
func: function() {
// 运行用户代码
const user_code = getUserCode();
GM_addElement(document.head, 'script', {
textContent: user_code
});
// 接收用户代码
GM_registerMenuCommand('自定义用户代码', function() {
const user_input = prompt('您可以在这里输入一些您自己的javascript代码,脚本会在页面开始加载时执行它', getUserCode());
if (user_input === null) { return; }
saveUserCode(user_input);
});
function getUserCode() {
const user_code = GM_getValue('user-code', '');
return user_code;
}
function saveUserCode(user_code) {
GM_setValue('user-code', user_code);
}
}
}]);
})();