JavaScript防流量劫持-前端安全_js流量劫持怎么操作_onlyhom的博客-程序员秘密

技术标签: 前端安全  js  防止流量劫持  

劫持产生的原因和方式

在网页开发的访问过程中,http是我们主要的访问协议。我们知道http是一种无状态的连接。即没有验证通讯双方的身份,也没有验证信息的完整性,所以很容易受到篡改。运营商就是利用了这一点篡改了用户正常访问的网页,插入广告或者其他一些杂七杂八的东西,达到盈利的目的。

运营商的一般做法有以下手段:

  • 对正常网站加入额外的广告,这包括网页内浮层或弹出广告窗口;
  • 针对一些广告联盟或带推广链接的网站,加入推广尾巴;
  • 把我们的站点非法解析到其他的站点,比如我们在浏览器输入http://baidu.com,百度绑定的服务器ip地址是111.13.101.208,此时如果运营商的dns服务器将baidu.com的对应的ip地址改为qq的服务器ip 14.17.32.211,我们输入http://baidu.com就会跳转到QQ的页面。

以上的手段,通过原理归纳为两种:

1、HTTP劫持

当我们使用HTTP请求请求一个网站页面的时候,网络运营商会在正常的数据流中插入精心设计的网络数据报文,让客户端(通常是浏览器)展示“错误”的数据,通常是一些弹窗,宣传性广告或者直接显示某网站的内容,大家应该都有遇到过。做法1、2就是通过这种方式

2、DNS劫持

我们通过域名访问网页的时候,都需要通过DNS服务器把域名解析到对应的服务器地址上,而用户上网的DNS服务器都是运营商分配的。所以,在这个节点上,运营商可以为所欲为。做法3就是通过这种方式

对于以上的劫持方式,我们作为前端的开发人员,通过javascript如何来做到有效的防护呢?

对于DNS劫持,由于发生在域名解析的时候,我们无法控制,javascript更无能为力。我们能做的就是拿起手机,投诉网络运营商,或者直接打工信部电话(12300)投诉。

http劫持防范

对于http劫持,运营商在实现上一般有以下几种做法

  • iframe嵌套展示原来正常网页
  • 在原html中插入js,再通过js脚本安插广告
  • 直接返回一个带广告的HTML

首先我们来看页面被嵌入了 iframe 的情况。网络运营商为了尽可能地减少植入广告对原有网站页面的影响,通常会通过把原有网站页面放置到一个和原页面相同大小的 iframe 里面去,那么就可以通过这个 iframe 来隔离广告代码对原有页面的影响。这种情况比较容易处理。我们只要判断我们的页面是否被嵌套在iframe中即可。Window对象中有两个属性self(指向本身的窗口),top(指向顶层的窗口)可以帮我们来识别判断

我们可以这样简单判断:

if (window.self != window.top) {
    
  var url = location.href;
    top.location = url;
}

但是,有时候我们在实际业务中,我们的页面确实需要被嵌套在iframe中推广,上面的判断会导致页面无法嵌套,这时候我们可以采用配置域名白名单的方式来解决:

var avoidIframeNest = {
    
    whiteList : [],
    init: function(whiteList){
    
        if(Object.prototype.toString.call(whiteList) == "[object Array]"){
    
            this.whiteList = whiteList;
        }
        this.redirect();
    },
    redirect: function(){
    
        if(self != top){
    
            var parentUrl = document.referrer;
            //是否在白名单内
            for(var i = 0 ,length = this.whiteList.length ; i < length ; ++ i){
    
                var reg = new RegExp(this.whiteList[i],'i');

                if(reg.test(parentUrl)){
    
                  return;
                }
            }
            //页面跳转
            var url = location.href;
            top.location = url;
        }
    }
}

通过配置白名单的方式,比较适合于我们经常用到的域名,通常我们会遇到这样的需求,合作方要求嵌套我们的页面,我们如果将合作方也加入到我们白名单,一方面会导致白名单很长,另一方面我们需要手动去改代码,这样很不方便。这种情况,我们可以在嵌套的url上加上域名的参数判断,要求嵌套页面带上域名参数,如果匹配,就认为合法。

var avoidIframeNest = {
    
    whiteList : [],
    init: function(whiteList){
    
        if(Object.prototype.toString.call(whiteList) == "[object Array]"){
    
            this.whiteList = whiteList;
        }
        this.redirect();
    },
    redirect: function(){
    
        if(self != top){
    
            var parentUrl = document.referrer;
            //是否在白名单内
            for(var i = 0 ,length = this.whiteList.length ; i < length ; ++ i){
    
                var reg = new RegExp(this.whiteList[i],'i');

                if(reg.test(parentUrl)){
    
                  return;
                }
            }

            //判断URL是否带指定参数
            var iframeDomain = this.getUrlParam('iframe_domain');
            if(iframeDomain && parentUrl.indexOf(iframeDomain) != -1){
    
                return;
            }
            //页面跳转
            var url = location.href;
            top.location = url;
        }
    },
    getUrlParam : function(key) {
    
        var regStr = "^.*[\\?|\\&]" + key + "\\=([^\\&]*)",
            url = location.href;
        reg = new RegExp(regStr,'i');;
        var ret = url.match(reg);
        if (ret != null) {
    
            return decodeURIComponent(ret[1]);
        } else {
    
            return "";
        }
    }
}

avoidIframeNest.init(['baidu.com']);

通过上述的方法,基本可以解决iframe嵌套问题

对于js注入问题,一般都会在页面中插入图片标签,展示广告,诱导用户点击。针对这种方式,我们可以通过监控页面插入的图片内容来检测。这里,我们可以利用HTML5的新特性MutationObserver 和window下的DOMNodeInserted事件

Mutation Observer(变动观察器)是监视DOM变动的接口。当DOM对象树发生任何变动时,Mutation Observer会得到通知。具体的介绍可以参考:

可以监听某个 DOM 范围内的结构变化

http://www.cnblogs.com/jscode/p/3600060.html

DOMNodeInserted顾名思义,可以监听某个 DOM 范围内的结构变化,这个特性只有在firefox的低版本和webkit中使用,IE不支持,这里我们可以作为低版本浏览器的兼容实现。

var validInsertImg = {
    
    httpReg : /^http:\/\/(.*\.baidu\.com|.*\.netwin\.com)\//,
    //验证非法图片
    validIllegalityImg : function(src){
    
        var httpReg = this.httpReg;
        return !httpReg.test(src);
    },
    init : function(){
    
        this.monitor();
    },
    monitor: function(){
    
        var MutationObserver = window.MutationObserver ||
        window.WebKitMutationObserver || 
        window.MozMutationObserver;
        var mutationObserverSupport = !!MutationObserver;
        //html5监控变化属性
        if(!mutationObserverSupport){
    
            this.mutationListen(MutationObserver);
        }else{
    
            this.insertedListen();
        }
    },
    insertedListen : function(){
    
        var that = this;
        document.addEventListener('DOMNodeInserted', function(e) {
    
            var dom = e ? e.srcElement : document.documentElement;
            if (!dom.outerHTML) {
    
                return;
            }
            var imgList = (dom.nodeName.toUpperCase() == 'IMG') ? [dom] : dom.getElementsByTagName('img');
            if (!imgList || imgList.length == 0) {
    
                return;
            }
             for (var i = 0; i < imgList.length; i++) {
    
                   that.removeNode(imgList[i]);
            }
        });
    },
    mutationListen: function(MutationObserver){
    
        var that = this;
        var observer = new MutationObserver(function(mutations){
    
            mutations.forEach(function(mutation){
    
                var nodes = mutation.addedNodes;
                for(var i = 0 ; i < nodes.length ; i++){
    
                    var node = nodes[i];
                    that.removeNode(node);
                }
            })
        })
        observer.observe(document, {
    
          subtree: true,
          childList: true
        });

    },

    //删除node
    removeNode : function(node){
    
        if(node.nodeName.toUpperCase() == 'IMG'){
    
            var src = node.src;
            if(this.validIllegalityImg(src)){
    
                node.parentNode.removeChild(node);
                  console.log('拦截可疑静态脚本:', node.src);
            }
        }
    }
}

validInsertImg.init();

body = document.getElementsByTagName('body')[0];
 var img = document.createElement('img');
     img.setAttribute('src','http://m.baidu.com/img/b')
      body.appendChild(img);

 var img1 = document.createElement('img');
     img1.setAttribute('src','/YTRYTRY/A.PNG')
     body.appendChild(img1);

对于在返回html内容中插入广告,我们可以借鉴注入的方式,进入页面就检测的img图片路径是否在白名单内

以上方法,都是针对运营商劫持的常用手段进行的一些黑科技操作。只能尽量的减少劫持给我们带来的负面影响。针对劫持问题,最好的办法就是全站升级https的方式,验证通讯双方的身份以及信息的安全性。

但是https也不能完全的解决劫持问题,如果https页面被劫持,浏览器会出现空白页面或者提示不安全,无法显示正常的内容。这也会影响到用户的体验。但是还是推荐使用https,如果大部分的网站都使用了https,运营商的劫持无法达到目的,自然不会去做这样吃力不讨好的事情。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/oulihong123/article/details/94380938

智能推荐

推荐系统——用户画像_Sin_Geek的博客-程序员秘密

写在最开始,本人不是专业做推荐系统的,本文只是整理最近培训的所见所得。学生生涯的时候略微了解些推荐系统的皮毛,工作后暂时还没做这方面的工作,但之后的工作规划会涉及到相关领域知识,另加上本人对算法类培训颇感兴趣,于是参加了近期的培训,自此整理消化下相关知识,聊以慰藉。目前推荐系统的用户画像都是对用户的向量化表示,推荐系统的用户画像是给计算机用的,而不是给人看的。用户画像不是推荐系统的目的,而是在构...

如何设计网站?_cukexr2833的博客-程序员秘密

How to design the site? In this article we will look at such aspects of website designing as: choosing an editor for preparing the site layout, defining sizes of the site, how to save sources, how to...

stellar恒星网络安装_倒不了的塔的博客-程序员秘密

一、postgersql安装配置参考资料: https://www.postgresql.org/download/linux/ubuntu/sudo sh -c 'echo &quot;deb http://apt.postgresql.org/pub/repos/apt/ $(lsb_release -cs)-pgdg main&quot; &amp;gt; /etc/apt/sources.list.d/pgdg...

kmdf驱动教程2——驱动程序与应用程序通信_hustd10的博客-程序员秘密

1 简述作为写在最前面的话,我不会在本教程中专门去介绍KMDF框架、对象、方法等基础知识,因为相关资料已经不少,而且文档中都有,如果读者不了解,应该先去看看相关内容。驱动程序作为内核模式的一部分,都是为我们的应用程序服务的。而我们在教程1中编写的HelloWord不仅没有为我们服务,而且我们还难以控制,除了在设备管理器中操作,连打印出来的信息都要特殊的工具才能看得到。所以,一方面我们要让驱动程序为我

vue element-ui 绑定@keyup.enter事件无效解决办法_elementui keyup事件_船长在船上的博客-程序员秘密

一般情况解决办法:在按钮上绑定keyup事件,加上.native覆盖原有封装的keyup事件&lt;el-button type="primary" class="sub-btn" style="width:100%;" :disabled="btnDisabled" :loading="logining" @click.native.prevent="handleSubmit" @keyup.native.enter="handleSubmit"&gt;{{loginStr}}&lt;/el-bu

JavaScript调用pc和手机摄像头_万花果子的博客-程序员秘密

主要是pc和手机的浏览器可以调用,亲测可用,但是微信扫二维码则不可以,微信的话应该还是要调微信接口<!DOCTYPE html> <html lang="zh-CN"> <head> <meta charset="UTF-8"> <title>GetUserMedia实例</title> </head> <body> <video id="vide

随便推点

ORA-08002: 序列XXXX尚未在此会话中定义_xafc的博客-程序员秘密

在pl/sql的sql窗口执行如下语句时,报ORA-08002错误。select seq_test.currval from dual;Solution Description:---------------------The NEXTVAL function acts as a sequence initializer. This can be misleading si...

今天遇到安装网络打印机报错提示windows 无法连接 0x00000bcb_旭旭的记录的博客-程序员秘密

场景直接在地址栏中输入对方的地址,看到共享打印机,然后直接点连接,安装驱动到一半就报错“windows 无法连接 0x00000bcb“鼓捣了半天也没找到具体原因。后来在网上找到一个说是补丁问题,就直接打开程序-卸载-win更新,从上到下一个个删除,删除一个就连接一下打印机,删到KB4022722时,再连接打印机直接通过了。然后再重新打补丁,一切使用正常。在这做个记录。另外就是现在好多网上银行升级了,也有一些地方行没有升级,都调用IE。对于这样的应该尽可能装32位的系统。本身硬件不支持64装64的没

Echarts实现柱状图闪烁效果_echarts柱状图闪烁_橙子微笑的博客-程序员秘密

Echarts实现柱状图闪动效果实现效果如下:echarts中没有实现闪烁效果功能,但是我们可以通过setInterval()来实现闪烁效果代码如下:```javascript&lt;template&gt; &lt;div class="stratop"&gt; &lt;div class="topTitle" &gt; &lt;svg-icon icon-class="alarun" style="color:#5cd9e8; "&gt;&lt;/svg-icon&gt;

数字证书结构描述+解析的C程序设计和实现_数字证书结构体设计_Runner1st的博客-程序员秘密

文章目录1 X.509 证书结构描述1.1 整体结构1.2 证书内容1.3 编码2 源代码3 编译运行结果1 X.509 证书结构描述1.1 整体结构证书内容、签名算法、签名结果。tbsCertificateTBSCertificatesignatureAlgorithmAlgorithmIdentifiersignatureValueBIT STRING...

函数 | Python内置函数详解—数学运算类_看那个码农的博客-程序员秘密

本篇推文共计500个字,阅读时间约1分钟。本期推文是Python系列教程的第九期前八期我们讲了有关Python的发展历史:Python基础20讲 | 深扒码农那些不为人知的事编码特点:编...

推荐文章

热门文章

相关标签