如何使用 jest 和 lint-staged 只檢測發生改動的文件

  • 2021 年 6 月 23 日
  • 筆記

我們現在在推進 EPC 的過程中,單元測試是必備的技能,在本地的 Git commit 之前進行單測非常有必要,總不能把所有的單測的壓力都放在流水線上。

畢竟在流水線運行單測的成本還是挺高的,從 push 上去觸發流水線,到感知單測的結果,至少需要好幾分鐘的時間。

棒棒噠-前端小茶館公眾號

因此我們有必要在 git commit 進行一些單測的檢測。不過若我們每次在 commit 之前都完整地運行所有的單測用例,一個是沒必要,再一個是耗時很長。

那應該怎麼只運行有變動的文件的單測用例呢?

1. 使用 husky 和 lint-staged

我們接下來要使用 husky 和 lint-staged 組件,來實現在 commit 之前檢測只發生變動的文件。

  • husky可以讓我們很方便地設置 pre-commit 的鉤子;
  • lint-staged組件能夠在 Git commit 提交之前,獲取上次 commit 到現在所有發生變動的文件。我們可以利用這個特性來運行 jest;

1.1 配置 husky 和 lint-staged

首先我們來安裝和配置這兩個組件。

$ npm i husky lint-staged --save-dev
$ npm set-script prepare "husky install"
$ npm run prepare
$ npx husky add .husky/pre-commit "npx lint-staged"

運行完上述 4 個命令後,然後在package.json中配置 lint-staged:

{
  "scripts": {
    "test:staged": "jest --bail --findRelatedTests"
  },
  "lint-staged": {
    "src/**/*.{js,jsx,ts,tsx}": ["npm run test:staged"]
  }
}

在 lint-staged 中支援node-glob通配符的配置,同時也支援配置多個,若發生變動的文件路徑滿足配置,則觸發後面的命令。

上面通配符的意思是:src 目錄中任意路徑的任意以.js 或.jsx 或.ts 或.tsx 結尾的文件。

拽拽的-前端小茶館公眾號

1.2 配置 jest

我們在上面的 test:staged 命令配置上了 jest。

  • bail: 只要遇到運行失敗的單測用例即退出;
  • findRelatedTests: 檢測指定的文件路徑;

其他更多的參數,可以直接查閱官方文檔Jest CLI Options

很多公共的數據,我們可以直接在jest.config.js中進行配置:

module.exports = {
  roots: ['<rootdir>/src'], // 查找src目錄中的文件
  collectCoverage: true, // 統計覆蓋率
  coverageDirectory: 'coverage', // 覆蓋率結果輸出的文件夾
  coverageThreshold: {
    // 所有文件總的覆蓋率要求
    global: {
      branches: 60,
      functions: 60,
      lines: 60,
      statements: 60,
    },
    // 匹配到的單個文件的覆蓋率要求
    // 這裡也支援通配符的配置
    './src/**/*.{ts,tsx}': {
      branches: 40,
      functions: 40,
      lines: 40,
      statements: 40,
    },
  },
  // 匹配單測用例的文件
  testMatch: ['<rootdir>/src/**/__tests__/**/*.{js,jsx,ts,tsx}', '<rootdir>/src/**/*.{spec,test}.{js,jsx,ts,tsx}'],
  // 當前環境是jsdom還是node
  testEnvironment: 'jsdom',
  // 設置別名,若不設置,運行單測時會不認識@符號
  moduleNameMapper: {
    '^@/(.*)$': '<rootdir>/src/$1',
  },
};

上面兩個配置完成後,當本次 commit 發生變動的文件滿足要求,至少有 1 個文件滿足時,則就會執行npm run test:staged

我們先運行下所有的測試用例:

npm run test-前端小茶館公眾號

可以看到,我們實際上有 2 個源文件,3 個測試文件。不過 add 相關的已經在上次 commit 提交過了,本次提交時,只有 uitils 和 utils.test 有變動。

$ git add .
$ git ci -m 'test(utils): only test utils changed file'

從給出的測試報告能看出來,當前只檢測了發生變動的 utils 文件:

test:staged-前端小茶館公眾號

2. 覆蓋率的要求

我們在上面通過 jest.config.js 中的coverageThreshold屬性,設置了全局覆蓋率和單個文件的覆蓋率。

我們再在程式碼新增幾個文件,但不配置對應的測試文件。然後運行時發現,如果沒有對應的測試文件,就不會檢查該文件的覆蓋率。

我這裡特意把概率設置 100%,然後 math.js 沒有對應的測試文件:

覆蓋率-前端小茶館公眾號

從運行的測試結果來,這裡只檢測了有測試文件的 utils.js,並沒有檢測到 math.js。

這裡我們就要新增一個屬性了collectCoverageFrom

{
  collectCoverageFrom: ["src/**/*.{js,jsx,ts,tsx}"],
}

這時,我們再運行單測時,就能把所有符合要求的文件,都納入到覆蓋率的考核里了。

nice-前端小茶館公眾號

3. collectCoverageFrom 的坑

我們在使用 commit 提交時,會觸發lint-staged,按我們在第 1 節說的,應該只運行發生變動的實例,覆蓋率也只輸出當前運行的實例。

但實際上並不是如此,若配置了collectCoverageFrom,無論是怎樣運行單測,他都會輸出所有符合要求的文件的覆蓋率數據:

lint-staged與collectCoverageFrom-前端小茶館公眾號

從黃色框框中可以看到,我們本次只提交了 utils.js,按說應該只運行和計算 utils.js 的單測覆蓋率即可,但實際上會把所有的覆蓋率都輸出出來,然後大部分數據為 0,無法滿足設置的覆蓋率的要求。

但我們又不能不設置 collectCoverageFrom 屬性,最後我的解決辦法是:排除法

{
  collectCoverageFrom: ['!src/**/*.d.ts', '!src/**/*{.json,.snap,.less,.scss}'],
}

這樣我們在 commit 提交時,就能滿足只從被檢測的文件中提取覆蓋率。

4. 總結

工欲興其事,必先利其器。當我們提前把配置搭建完成後,就可以進行開發啦。

王者嗎-前端小茶館公眾號

歡迎關注我的公眾號:「前端小茶館」。

前端小茶館公眾號