Article Image
read

使用 Git 的時候常常會遇到一個狀況:commit 之後才發現 jshint 檢查不通過,像是少了個分號之類。當這是一個多人合作專案時,merge 之後一整排的 jshint 錯誤那更是歡樂無比。

#解決方式

###懶人包 || TL;DR Git Pre-commit Hook + JSHint (or JSLint 或任何你在用的檢查工具)

###系統需求 須先安裝 JSHint

$ npm install -g jshint

###詳細說明及設定步驟 既然我們需要 commit 時後都能通過 jshint 檢查,那麼就可以使用 Git Hook 來完成這件任務。Git Hook 簡單的說就是一個 trigger:當某 git 動作發生的時候,會去執行對應的 hook。以本文來說,我們需要在 commit 之前,執行一個幫我們做 jshint 檢查的 shell script。

設定方式非常簡單,到專案底下的 .git/hooks 裏面,建立一個名為 pre-commit 的檔案,檔案的主要任務當然就是執行 jshint:

#!/bin/sh
#
# Run JSHint validation before commit.

files=$(git diff --cached --name-only --diff-filter=ACM | grep ".js$")
pass=true

echo "files:"
if [ "$files" != "" ]; then
    for file in ${files}; do
        echo $file
        result=$(jshint -c .jshintrc ${file})

        if [ "$result" != "" ]; then
            echo $result
            pass=false
        fi
    done
fi


if $pass; then
    exit 0
else
    echo ""
    echo "COMMIT FAILED:"
    echo "Some JavaScript files are invalid. Please fix errors and try committing again."
    exit 1
fi

這個 shell script 利用 git diff 把這次 commit 修改的檔案名稱抓出來,並以 grep 過濾出 .js 檔,然後對他們執行 jshint。(在這邊我有設定 jshint 的設定檔為 .jshintrc )如果沒有問題就 exit 0 正常結束,git commit 就會繼續執行,如果有問題就印出錯誤訊息並 exit 1 中斷 commit。

#Grunt 進階版

如果你的專案有很多人在用,又剛好是使用 grunt,那麼可以新增下列 task 幫你自動安裝 git hook。

grunt.registerTask('install-hooks', function () {
  var fs = require('fs');

  // my precommit hook is inside the repo as /pre-commit.hook
  // copy the hook file to the correct place in the .git directory
  grunt.file.copy('pre-commit.hook', '.git/hooks/pre-commit');

  // chmod the file to readable and executable by all
  fs.chmodSync('.git/hooks/pre-commit', '755');
});

記得把你的 pre-commit 檔放在 repo 的 /pre-commit.hook,這樣使用這個 repo 的人只要執行一次 grunt install-hooks,系統就會自動安裝我們寫好的 hooks。

當然 hooks 有很多種,pre-commit hook 也可以做更多事情,就看個人需求更改。

Blog Logo

Fin Chen


Published

Image

Fin's Nest

Things about web develop

Back to Overview