js window.onerror 錯誤監聽,發到后台


 

window.onerror = function (message, url, lineNo, columnNo, error)
參數信息

1、message {String} 錯誤信息。直觀的錯誤描述信息,不過有時候你確實無法從這里面看出端倪,特別是壓縮后腳本的報錯信息,可能讓你更加疑惑。

2、url {String} 發生錯誤對應的腳本路徑,比如是你的http://a.js報錯了還是http://b.js報錯了。

3、lineNo {Number} 錯誤發生的行號。

4、columnNo {Number} 錯誤發生的列號。

5、error {Object} 具體的 error 對象,包含更加詳細的錯誤調用堆棧信息,這對於定位錯誤非常有幫助。

注意

1、對於跨域的JS資源,window.onerror拿不到詳細的信息,需要往資源的請求添加額外的頭部。
靜態資源請求需要加多一個Access-Control-Allow-Origin頭部,也就是需要后台加一個Access-Control-Allow-Origin,同時script引入外鏈的標簽需要加多一個crossorigin的屬性。這樣就可以獲取准確的出錯信息。

為了方便設置Header,可以把error.js做一個小改動,更名為:error-js.php

<?php 
  header('Access-Control-Allow-Origin:*'); 
  header('Content-type:text/javascript'); 
?> 
throw new Error('出錯了');


2、壓縮之后的代碼,我們得到錯誤的信息,但是我們卻無法定位到錯誤的行數

實例1:
<!DOCTYPE html>
<html>
<head>
    <title>Js錯誤捕獲</title>
    <script type="text/javascript">
        var MaxErrorReportLimit = 10;
        window.onerror = function(msg,url,line,col,error){
            // 同一個頁面最多上報10次錯誤,防止某個循環錯誤頁面一直打開,不斷的報錯
            if (MaxErrorReportLimit-- < 0) return;
            //沒有URL不上報!上報也不知道錯誤
            if (msg != "Script error." && !url){
                return true;
            }
            //采用異步的方式
            //我遇到過在window.onunload進行ajax的堵塞上報
            //由於客戶端強制關閉webview導致這次堵塞上報有Network Error
            //我猜測這里window.onerror的執行流在關閉前是必然執行的
            //而離開文章之后的上報對於業務來說是可丟失的
            //所以我把這里的執行流放到異步事件去執行
            //腳本的異常數降低了10倍
            setTimeout(function(){
                var data = {};
                //不一定所有瀏覽器都支持col參數
                col = col || (window.event && window.event.errorCharacter) || 0;

                data.url = url;
                data.line = line;
                data.col = col;
                if (!!error && !!error.stack){
                    //如果瀏覽器有堆棧信息
                    //直接使用
                    data.msg = error.stack.toString();
                }else if (!!arguments.callee){
                    //嘗試通過callee拿堆棧信息
                    var ext = [];
                    var f = arguments.callee.caller, c = 3;
                    //這里只拿三層堆棧信息
                    while (f && (--c>0)) {
                        ext.push(f.toString());
                        if (f  === f.caller) {
                            break;//如果有環
                        }
                        f = f.caller;
                    }
                    ext = ext.join(",");
                    data.msg = error.stack.toString();
                }
                //把data上報到后台!
                console.log(data,'======data=====')
                var xhr=new XMLHttpRequest();
                xhr.open('POST','/get_error.php',false);
                // 添加http頭,發送信息至服務器時內容編碼類型
                xhr.setRequestHeader('Content-Type','application/x-www-form-urlencoded');
                // xhr.setRequestHeader('Content-Type','application/json');
                xhr.onreadystatechange=function(){
                    if (xhr.readyState==4){
                        if (xhr.status==200 || xhr.status==304){
                            // console.log(xhr.responseText);
                        }
                    }
                }
                xhr.send(JSON.stringify(data));
            },0);
            var show_js_error = true;
            if(show_js_error != true){
                //關閉js報錯提示
                return true;
            }

        };
    </script>
</head>
<body>
<!--<script type="text/javascript" src="http://127.0.0.1/error.js"></script>-->
<script type="text/javascript" >
    aa      //故意報錯測試
    // throw new Error('出錯了');
</script>
</body>
</html>

get_error.php

<?php
echo '<pre>';
print_r($_REQUEST);

 

實例2(引入js插件):

<!DOCTYPE html>
<html>
<head>
    <title>Js錯誤捕獲</title>
</head>
<body>
<script type="text/javascript" src="badJsReport.js"></script>
<SCRIPT>
    badJsReport({
        url:'get_error.php', //發送到后台的url *必須
    })
    aa       //故意報錯測試
    // throw new Error("出錯了!");
</SCRIPT>
</body>
</html>

badJsReport.js

/**
 * Name:    badJsReport.js
 * Version  1.1.0
 * Author   xiangyulaodi
 * Address: https://github.com/xianyulaodi/badJsReport
 * Released on: December 22, 2016
 */

;(function(){

    'use strict';

    if (window.badJsReport){ 

       return window.badJsReport 
    };

    /*
    *  默認上報的錯誤信息
    */ 
    var defaults = {
        msg:'',  //錯誤的具體信息
        url:'',  //錯誤所在的url
        line:'', //錯誤所在的行
        col:'',  //錯誤所在的列
        error:'', //具體的error對象
    };

    /*
    *ajax封裝
    */
    function ajax(options) {
        options = options || {};
        options.type = (options.type || "GET").toUpperCase();
        options.dataType = options.dataType || "json";
        var params = formatParams(options.data);

        if (window.XMLHttpRequest) {
           var xhr = new XMLHttpRequest();
        } else { 
           var xhr = new ActiveXObject('Microsoft.XMLHTTP');
        }

        xhr.onreadystatechange = function () {
           if (xhr.readyState == 4) {
               var status = xhr.status;
               if (status >= 200 && status < 300) {
                   options.success && options.success(xhr.responseText, xhr.responseXML);
               } else {
                   options.fail && options.fail(status);
               }
           }
        }

        if (options.type == "GET") {
           xhr.open("GET", options.url + "?" + params, true);
           xhr.send(null);
        } else if (options.type == "POST") {
           xhr.open("POST", options.url, true);
           //設置表單提交時的內容類型
           xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded");
           xhr.send(params);
        }
    }

    /*
    *格式化參數
    */
    function formatParams(data) {
       var arr = [];
       for (var name in data) {
           arr.push(encodeURIComponent(name) + "=" + encodeURIComponent(data[name]));
       }
       arr.push(("v=" + Math.random()).replace(".",""));
       return arr.join("&");
    }


    /*
    * 合並對象,將配置的參數也一並上報
    */
    function cloneObj(oldObj) { //復制對象方法
      if (typeof(oldObj) != 'object') return oldObj;
      if (oldObj == null) return oldObj;
      var newObj = new Object();
      for (var prop in oldObj)
        newObj[prop] = oldObj[prop];
      return newObj;
    };

    function extendObj() { //擴展對象
      var args = arguments;
      if (args.length < 2) {return;}
      var temp = cloneObj(args[0]); //調用復制對象方法
      for (var n = 1,len=args.length; n <len; n++){
        for (var index in args[n]) {
          temp[index] = args[n][index];
        }
      }
      return temp;
    }

   /**
   * 核心代碼區
   **/
   var badJsReport=function(params){
       var MaxErrorReportLimit = 10;
      if(!params.url){return}
      window.onerror = function(msg,url,line,col,error){
          // 同一個頁面最多上報10次錯誤,防止某個循環錯誤頁面一直打開,不斷的報錯
          if (MaxErrorReportLimit-- < 0) return;
          //采用異步的方式,避免阻塞
          setTimeout(function(){

              //不一定所有瀏覽器都支持col參數,如果不支持就用window.event來兼容
              col = col || (window.event && window.event.errorCharacter) || 0;

              defaults.url = url;
              defaults.line = line;
              defaults.col =  col;

              if (error && error.stack){
                  //如果瀏覽器有堆棧信息,直接使用
                  defaults.msg = error.stack.toString();

              }else if (arguments.callee){
                  //嘗試通過callee拿堆棧信息
                  var ext = [];
                  var fn = arguments.callee.caller;
                  var floor = 3;  //這里只拿三層堆棧信息
                  while (fn && (--floor>0)) {
                     ext.push(fn.toString());
                     if (fn  === fn.caller) {
                          break;//如果有環
                     }
                     fn = fn.caller;
                  }
                  defaults.msg = ext.join(",");
                }
                // 合並上報的數據,包括默認上報的數據和自定義上報的數據
                var reportData=extendObj(params.data || {},defaults);
                
                // 把錯誤信息發送給后台
                ajax({
                    url: params.url,      //請求地址
                    type: "POST",         //請求方式
                    data: reportData,     //請求參數
                    dataType: "json",
                    success: function (response, xml) {
                        // 此處放成功后執行的代碼
                      params.successCallBack&&params.successCallBack(response, xml);
                    },
                    fail: function (status) {
                        // 此處放失敗后執行的代碼
                      params.failCallBack&&params.failCallBack(status);
                    }
                 });

          },0);

          return true;   //錯誤不會console瀏覽器上,如需要,可將這樣注釋
      };

  }
    
  window.badJsReport=badJsReport;

})();

/*===========================
badJsReport AMD Export
===========================*/
if (typeof(module) !== 'undefined'){
    module.exports = window.badJsReport;
}
else if (typeof define === 'function' && define.amd) {
    define([], function () {
        'use strict';
        return window.badJsReport;
    });
}
View Code

get_error.php

<?php
echo '<pre>'; print_r($_REQUEST);

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM