99%的程式都沒有考慮的網路異常

来源:https://www.cnblogs.com/qcloud1001/archive/2018/12/26/10180903.html
-Advertisement-
Play Games

本文由雲+社區發表 絕大多數程式只考慮了介面正常工作的場景,而用戶在使用我們的產品時遇到的各類異常,全都丟在看似 ok 的 try catch 中。如果沒有做好異常的相容和兜底處理,會極大的影響用戶體驗,嚴重的還會帶來安全和資損風險。 介面異常,通常可以分為以下三類: CGI 邏輯出錯。 如調用方入 ...


本文由雲+社區發表

絕大多數程式只考慮了介面正常工作的場景,而用戶在使用我們的產品時遇到的各類異常,全都丟在看似 ok 的 try catch 中。如果沒有做好異常的相容和兜底處理,會極大的影響用戶體驗,嚴重的還會帶來安全和資損風險。

介面異常,通常可以分為以下三類:

  • CGI 邏輯出錯。如調用方入參缺失類業務邏輯報錯;
  • 服務不穩定。如伺服器不穩定導致 nginx 各類 500、502,cgi 路徑調整導致的 404
  • 用戶網路環境差。如,網路不穩定、網速慢、運營商劫持等

那麼,我們在寫代碼時,如何快速的模擬這些介面異常,做好程式的相容處理呢?

今天向大家介紹網路調試神器 whistle 的網路異常調試方法,如果你還沒用過 whistle,請參考《8102 年的程式員不需要 Hosts 和 Fiddler》。

假設我們有以下前端頁面 index.html,放置在自己的本地路徑:

<p id="success" style="color:green;"></p>
<p id="fail" style="color:red;"></p>
<script>
  fetch(`/mock?r=${Math.random()}`)
    .then(response => {
      return response.json()
    })
    .then(v => {
      document.getElementById('success').innerHTML = v.data;
    }).catch(err => {
      document.getElementById('fail').innerHTML = err.message;
    })
</script>

接下來,打開 whistle Rules 配置面板 http://127.0.0.1:8899/#rules ,配置模擬的 demo page 和 mock CGI:

*/mock file://({"code":0,"data":"success"}) # 配置 mock cgi 為模擬的 json 數據
example.com file:///Users/kaiye/Projects/Markdown/20181213/ # 配置任意功能變數名稱到本地 demo 目錄,這裡註意替換成自己的路徑

打開 http://example.com ,正常邏輯下頁面展示出了綠色的 success ,現在我們開始加入一些網路異常。

1、業務邏輯異常處理

例如 CGI 沒有返回 data 欄位,而是返回了一個錯誤碼 code 和對應的 message,針對這種業務邏輯異常我們只需在第二個 then 中做好 code 值的判斷即可(註意,這裡的 code、message、data 只是示例,實際業務 CGI 中的 JSON 結構體的欄位名很可能不同):

fetch(`/mock?r=${Math.random()}`)
  .then(response => response.json())
  .then((v) => {
    // 業務邏輯異常處理
    if (v.code !== 0) {
      return Promise.reject(new Error(`ERROR_LOGIC_CODE:${v.code}`));
    }
    document.getElementById('success').innerHTML = v.data;
  })
  .catch((err) => {
    document.getElementById('fail').innerHTML = err.message;
  });

相應的 whistle 配置如下:

*/mock file://({"code":12345,"message":"some_logic_error"}) # 模擬業務邏輯異常

2、伺服器異常處理

如果伺服器直接拋出了 502 錯誤碼,我們希望代碼能給用戶提示的同時,再做一個異常上報。

fetch(`/mock?r=${Math.random()}`)
  .then((response) => {
    // 伺服器異常處理
    if (response.ok) {
      return response.json();
    }
    return Promise.reject(new Error(`ERROR_STATUS_CODE:${response.status}`));
  })
  .then((v) => {
    // 業務邏輯異常處理
    if (v.code !== 0) {
      return Promise.reject(new Error(`ERROR_LOGIC_CODE:${v.code}`));
    }
    document.getElementById('success').innerHTML = v.data;
  })
  .catch((err) => {
    const [type, value] = err.message.split(':');
    // 異常類型上報
    console.log(type, value);
    document.getElementById('fail').innerHTML = err.message;
  });

通過 whistle 的模擬配置如下:

*/mock statusCode://502 # 模擬 HTTP 狀態碼異常

3、介面被劫持註入

如果 CGI 被運營商劫持註入,可能導致介面返回一個不合法的 JSON 結構,最前面的 response.json() 會拋異常,我們可以提前 catch 住:

fetch(`/mock?r=${Math.random()}`).then((response) => {
  // 伺服器異常處理
  if (response.ok) {
    return (
      response
        .json()
        // 介面數據解碼異常處理
        .catch(err => Promise.reject(new Error('ERROR_DECODE_JSON')))
    );
  }
  return Promise.reject(new Error(`ERROR_STATUS_CODE:${response.status}`));
});

whistle 模擬配置如下:

*/mock file://(<div>hijacking</div>{"code":0,"data":"success"}) # 模擬介面被劫持註入 1

藉助 htmlAppendvalues 配置,可以模擬更複雜的註入示例:

*/mock file://({"code":0,"data":"success"}) htmlAppend://{hijacking.html} # 模擬介面被劫持註入 2
```hijacking.html
<script>
alert('hijacking')
</script>
```

4、用戶網路不穩定

如果我們要模擬請求發出 10 秒後斷網或網路不通的情況,可以通過 whistle 這樣配置:

*/mock reqDelay://10000 enable://abort # 模擬 10 秒超時後網路不通

讓用戶苦苦等待 10 秒,再報錯的體驗太糟糕。我們可以封裝一個能配置超時時間的請求發送函數,同時把上面提到的錯誤異常都一起配置進來。

<p id="success" style="color:green;"></p>
<p id="fail" style="color:red;"></p>
<script>
  function myFetch(url, configOptions) {
    const options = Object.assign(
      {
        timeout: 3000
      },
      configOptions
    )
    const { timeout } = options
    return new Promise((resolve, reject) => {
      // 超時異常處理
      const timer = setTimeout(() => {
        reject(new Error(`ERROR_TIMEOUT:${timeout}`))
      }, timeout)
      fetch(url, options)
        .then(data => {
          clearTimeout(timer)
          resolve(data)
        })
        .catch(err => {
          clearTimeout(timer)
          reject(err)
        })
    })
      .then(response => {
        // 伺服器異常處理
        if (response.ok) {
          return (
            response
              .json()
              // 介面數據解碼異常處理
              .catch(err => Promise.reject(new Error('ERROR_DECODE_JSON')))
          )
        } else {
          return Promise.reject(
            new Error(`ERROR_STATUS_CODE:${response.status}`)
          )
        }
      })
      .then(v => {
        // 業務邏輯異常處理
        if (v.code !== 0) {
          return Promise.reject(new Error(`ERROR_LOGIC_CODE:${v.code}`))
        } else {
          return v.data
        }
      })
      .catch(err => {
        const [type, value] = err.message.split(':')
        // 異常類型上報
        console.log(type, value)
        return Promise.reject(err)
      })
  }
  myFetch(`/mock?r=${Math.random()}`)
    .then(data => {
      document.getElementById('success').innerHTML = data
    })
    .catch(err => {
      document.getElementById('fail').innerHTML = err.message
    })
</script>

這樣,自定義的 myFetch 只需關註業務具體邏輯,針對不同的 catch error 做對應的處理。

除以上提到的協議命令字外,whistle 還支持 resSpeed 用於模擬低網速傳輸(單位:kb/s),tpl 協議則可以根據請求傳入參數來動態模擬不同的數據。在 Frames 面板,還可以對 WebSocket/Socket 請求進行暫停、延遲等網路異常的模擬。

小程式 fetch API 實現

最後,留一道思考題。

近來微信小程式開發非常火,小程式原生提供的 wx.request API 能用於發送 HTTPS 請求,請在它的基礎之上進行封裝,支持 promise 調用和 timeout 超時時間定義(小程式預設的請求超時定義在 app.json 中,不夠靈活),並針對以上提到的 HTTP 狀態碼異常、介面劫持註入、慢網路、無網路狀態等各種網路異常進行相容處理。

歡迎留言分享你的代碼實現

此文已由作者授權騰訊雲+社區發佈



您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.實現一個函數clone,可以對JavaScript中的5種主要的數據類型(包括Number、String、Object、Array、Boolean)進行值複製 2.下麵這個ul,如何點擊每一列的時候alert其index?(閉包) 3.對於MVVM的理解? 4.什麼是vue生命周期? 5.Vue ...
  • 請選擇地區: ...
  • 頂部: $(".back_top").click(function () { scrollTo(0, 0); }); function goTop() { $('html, body').animate({scrollTop:0}, 'slow'); } 底部: $(".back_bottom"). ...
  • 筆者相信在未來的某一天,Koa一定會完全替代Express,成為Node.js Web框架的主流。 ...
  • __proto__ 、prototype、constructor ...
  • 1.getter與setter javascript的大部分東西是簡化了的。這裡我們假設有一個student對象,並且我們用student.age來訪問它的age屬性,如果此時定義了age屬性,我們就會得到它 的值,如果沒有,我們就會得到undefined。 但是我們也可以編寫自定義的getter和 ...
  • 在初步完成了 "線上流程圖編輯工具" 之後,又接到了線上搭建頁面工具的需求,剛開始其實並不想接項目,因為從歷史以及現實原因來看,個性化及動態渲染都是很難解決的痛點,各種H5頁面搭建工具的不溫不火早已說明瞭這條路並沒有這麼好走,但從另一個方面來說,既然有了這樣的需求,那也就說明瞭現實的工作流程確實存在 ...
  • 頁面中經常會用到各式各樣的輪播圖,今天賀賀為大家介紹一種常用的方法,對於JS我們需要舉一反三,一種方法可以對多個輪播樣式進行渲染。 此種方法通過 JS 來改變 HTML 中標簽的 Class 名稱,從而達到輪播圖的效果; ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...