用js屏蔽脚本加载的广告

辛辛苦苦写了一个网站,可是用手机浏览器打开以后,在bottom的位置常常有广告显示,这是因为在wifi被dns劫持的状况下导入的广告脚本种子加载出的。

javascript

1.查找根源

知道病情才能对症下药,广告脚本种子基本上是动态建立script来加载脚本的:css

var ele;
if (/\.css[^\.]*$/.test(asset.url)) {
    ele = doc.createElement("link");
    ele.type = "text/" + (asset.type || "css");
    ele.rel = "stylesheet";
    ele.href = asset.url
}
else {
    ele = doc.createElement("script");
    ele.type = "text/" + (asset.type || "javascript");
    ele.src = asset.url
}

没错,document.createElement

java

2.对症下药

/** 1. 禁用动态添加脚本,防止广告加载 */
(function () {
    var createElement = document.createElement;
    document.createElement = function (tag) {
        switch (tag) {
            case 'script':
                console.log('禁用动态添加脚本,防止广告加载');
                break;
            default:
                return createElement.apply(this, arguments);
        }
    }
})();


3.思路扩展

可能会误伤友军,好比要加载百度的脚本等,当本身须要用到这个时,能够考虑监控script.src的值的变化来过滤脚本,可是经过定义各类setter规则来监听script都会报错,因此只能换一种思路:在script被添加前作处理。web

/** * 禁用动态添加脚本,防止广告加载 * * @param valid bool? true(valid)|false(invalid)|other(off) * @param rule array 配置容许(valid)|不容许(invalid)的脚本规则:支持regex、string、function */
(function(valid, rule) {
    if(typeof Element === 'undefined') console.log('IE8如下浏览器无效');
    var origin = new RegExp('^' + location.origin),Ele = Element;
    each(['appendChild', 'insertBefore', 'insertAfter'], proxy);

    function proxy(prop) {
        var proxy_obj = Ele.prototype[prop];
        Ele.prototype[prop] = function(elem) {
            if (!elem.children.length) {
                var tag = elem.tagName.toLowerCase();
                if (tag == 'script' && isBanScript(elem)) {
                    console.log('禁用脚本:' + elem.src);
                    var substitute = document.createElement('script');
                    substitute.innerHTML = '// 禁用脚本:' + elem.src;
                    elem = substitute;
                }
            }
            return proxy_obj.apply(this, arguments);
        };
    }

    function isBanScript(script) {
        if (origin.test(script.src)) return false;
        return valid === each(rule, match);

        function match(val) {
            var type = typeof val;
            if (type === 'string') {
                if (script.src == val) return true;
            } else if (type === 'function') {
                if (val(script)) return true;
            } else {
                if (val.test(script.src)) return true;
            }
            return false;
        }
    }

    function each(arr, fn) {
        if (arr) {
            for (var i = 0, n = arr.length; i < n; i++) {
                if (fn.call(arr[i], arr[i], i) === true) return false;
            }
        }
        return true;
    }
})(true, []);
//表示有效的脚本规则列表