草庐IT

Git Hooks简介及结合Husky和Commitlint检测提交代码规范

小洋人最happy 2023-04-08 原文

文章目录

一、Git Hooks

1.1 目标

在上篇文章中,我们使用git cz替代了git commit实现了规范化提交诉求,但是依然存在着有人会忘记使用的问题,那这篇文章就来看下,如何去解决这种问题?

在此之前,先来明确下最终实现的效果:

当《提交描述信息》不符合约定式提交规范时,阻止当前的提交,并抛出对应的错误提示

而实现这个目标,我们需要先了解一个概念,叫[Git hooks](https://git-scm.com/docs/githooks)(git 钩子、git毁掉方法),意思是指git在执行某个操作之前或之后进行一些其他额外的操作

而我们期望的阻止不合规的提交信息,就需要使用到hooks的钩子函数。

1.2 常用hooks

下面是整理的常用hooks

Git Hook调用时机说明
pre-applypathgit am执行前
applypath-msggit am执行前
post-applypathgit am执行后不影响git am结果
pre-commitgit commit执行前可以用git commit --no-verify绕过
cmmit-msggit commit执行前可以用git commit --no-verify绕过
post-commitgit commit执行后不影响git commit的结果
pre-merge-commitgit merge执行前可以用git merge --no-verify绕过
prepare-cmmit-msggit commit执行后,编辑器打开前
pre-rebasegit rebase执行前
post-checkoutgit checkout或git switch执行后如果不使用–no-checout参数,则在git clone 之后也会执行
post-mergegit commit执行后在执行git pull时也会被调用
pre-pushgit push执行前
pre-receivegit-receive-pack 执行前
post-rewritegit commit --amend或git rebase执行前
sendemail-validategit send-email执行前
updategit receive-pack执行后
post-receivegit receive-pack执行后不影响git-receive-pack的结果

1.3 核心钩子

在实际开发中,最常用的有两个:

Git Hook调用时机说明
pre-commitgit commit执行前,不接受任何参数,在获取提交日志消息并提交之前被调用可以通过git commit --no-verify绕过
cmmit-msggit commit执行前,可用于将消息规范化为某种项目标准格式,还可以用于在检查消息文件后拒绝提交可以通过git commit --no-verify绕过

简单来说:

  • commit-msg: 可以用来规划范标准格式,并且按需指定是否要拒绝本次提交
  • pre-commit:在提交前被调用,可以按需指定是否要拒绝本次提交

二、Commitlint

在上一节中,了解到git hooks的概念,那么接下来就是要git hooks去校验我们的提交信息。

首先介绍第一个工具commitlint,用来检查提交信息的。

2.1 安装

注:npm版本需要7.x及以上

在项目终端执行命令如下:

npm i @commitlint/config-conventional@12.1.4 @commitlint/cli@12.1.4 -D

2.2 创建配置文件

在项目根目录下,创建commitlint.config.js配置文件,内容如下:

module.exports = {
  // 继承的规则
  extends: ['@commitlint/config-conventional'],
  // 定义规则类型
  rules: {
    // type 类型定义,表示 git 提交的 type 必须在以下类型范围内
    'type-enum': [
      // 当前验证的错误级别
      2,
      // 在什么情况下进行验证
      'always',
      // 泛型内容,与cz-config.js中类型一致
      [
        'feat', // 新功能 feature
        'fix', // 修复 bug
        'docs', // 文档注释
        'style', // 代码格式(不影响代码运行的变动)
        'refactor', // 重构(既不增加新功能,也不是修复bug)
        'perf', // 性能优化
        'test', // 增加测试
        'chore', // 构建过程或辅助工具的变动
        'revert', // 回退
        'build' // 打包
      ]
    ],
    // subject 大小写不做校验
    'subject-case': [0]
  }
}

三、Husky

3.1 安装

安装依赖,如下:

npm install husky@7.0.1 -D

3.2 启动

启动hooks,生成.husky文件夹,执行命令如下:

npx husky install

执行完成后,在项目根目录下会生成.husky文件夹,如下:

3.3 生成指令并执行

package.json中配置prepare指令,如下:

"scripts": {
  ...
  "prepare": "husky install"
},

在终端窗口执行npm run prepare,如下:

3.4 通过commit-msg规范化提交信息

添加commitlinthookhusky中,在commit-msghooks下执行npx --no-install commitlint --edit "$1",完整命令如下:

npx husky add .husky/commit-msg 'npx --no-install commitlint --edit "$1"'

执行完成后,会发现.husky中多了commit-msg文件,如下:

3.5 通过pre-commit检测提交代码规范

3.4完成了强制规范化的提交要求,现在不符合规范的提交信息,将不能够再提交!

也许,聪明的你已经发现,还缺少一个规范化的处理,那就是代码格式提交规范处理

在前面的文章中介绍过Eslint + Prettier来对代码格式做规范,但这只能再本地做处理,并且还需要手动在VSCode中配置自动保存,那么问题就来了,要是有人忘记配置了怎么办?代码格式乱七八糟的直接提交了怎么办?

因此,我们需要有一种方式来规避这种风险,那就是使用Husky配合Eslint来完成。本质上就是通过husky监测pre-commit钩子,在该钩子下执行npx eslint --ext .js,.vue src指令进行相关检测。

在项目终端中执行指令,完成commit时的hook,如下:

npx husky add .husky/pre-commit "npx eslint --ext .js,.vue src"

执行完成后,会发现在.husky中多了一个文件:

四、验证测试

4.1 验证提交规范

通过上面操作,不符合规范的commit将不再可提交,开始测试

执行git commit -m "commit test",报错如下:

 xxx@xxx-MacBook453  ~/Projects/VSCodeProjects/xxx   master ✚  git commit -m "commit test" 
⧗   input: commit test
✖   subject may not be empty [subject-empty]
✖   type may not be empty [type-empty]

✖   found 2 problems, 0 warnings
ⓘ   Get help: https://github.com/conventional-changelog/commitlint/#what-is-commitlint

husky - commit-msg hook exited with code 1 (error)

至此,不符合规范的commit将不能够再提交。

4.2 验证代码规范

首先,先通过设置保存时自动格式化关闭,如下:

然后,修改代码,将单引号修改为双引号,如下:

export default {
  name: "HomeView",
  components: {
    HelloWorld
  }
}

执行git commit提交,报错如下:

说明,我们在提交时,对代码格式规范检测也成功了。

五、再进一步

通过前面几节的介绍,目前想要提交代码,就要保证代码格式规范提交信息格式规范,那是不是已经万事大吉了呢?

当然不是,世界上从来不缺的就是懒人,错误的代码格式可能会抛出很多的ESLint错误,让人看的头皮发麻,严重影响程序猿的幸福指数。

那有么有一种办法,让程序猿0配置的前提下,哪怕代码格式再乱,都可以自动帮助其修复对应的问题并完成提交呢?

必须有!!!

欲知后事如何,请看下章讲解~

有关Git Hooks简介及结合Husky和Commitlint检测提交代码规范的更多相关文章

  1. ruby-on-rails - 结合 meta_search 与 acts_as_taggable_on - 2

    我在开发的Rails3网站的一些搜索功能上遇到了一个小问题。我有一个简单的Post模型,如下所示:classPost我正在使用acts_as_taggable_on来更轻松地向我的帖子添加标签。当我有一个标记为“rails”的帖子并执行以下操作时,一切正常:@posts=Post.tagged_with("rails")问题是,我还想搜索帖子的标题。当我有一篇标题为“Helloworld”并标记为“rails”的帖子时,我希望能够通过搜索“hello”或“rails”来找到这篇帖子。因此,我希望标题列的LIKE语句与acts_as_taggable_on提供的tagged_with方法

  2. ruby - 如何在 buildr 项目中使用 Ruby 代码? - 2

    如何在buildr项目中使用Ruby?我在很多不同的项目中使用过Ruby、JRuby、Java和Clojure。我目前正在使用我的标准Ruby开发一个模拟应用程序,我想尝试使用Clojure后端(我确实喜欢功能代码)以及JRubygui和测试套件。我还可以看到在未来的不同项目中使用Scala作为后端。我想我要为我的项目尝试一下buildr(http://buildr.apache.org/),但我注意到buildr似乎没有设置为在项目中使用JRuby代码本身!这看起来有点傻,因为该工具旨在统一通用的JVM语言并且是在ruby中构建的。除了将输出的jar包含在一个独特的、仅限ruby​​

  3. ruby-on-rails - Rails 源代码 : initialize hash in a weird way? - 2

    在rails源中:https://github.com/rails/rails/blob/master/activesupport/lib/active_support/lazy_load_hooks.rb可以看到以下内容@load_hooks=Hash.new{|h,k|h[k]=[]}在IRB中,它只是初始化一个空哈希。和做有什么区别@load_hooks=Hash.new 最佳答案 查看rubydocumentationforHashnew→new_hashclicktotogglesourcenew(obj)→new_has

  4. ruby - RuntimeError(自动加载常量 Apps 多线程时检测到循环依赖 - 2

    我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("

  5. ruby-on-rails - 浏览 Ruby 源代码 - 2

    我的主要目标是能够完全理解我正在使用的库/gem。我尝试在Github上从头到尾阅读源代码,但这真的很难。我认为更有趣、更温和的踏脚石就是在使用时阅读每个库/gem方法的源代码。例如,我想知道RubyonRails中的redirect_to方法是如何工作的:如何查找redirect_to方法的源代码?我知道在pry中我可以执行类似show-methodmethod的操作,但我如何才能对Rails框架中的方法执行此操作?您对我如何更好地理解Gem及其API有什么建议吗?仅仅阅读源代码似乎真的很难,尤其是对于框架。谢谢! 最佳答案 Ru

  6. ruby - 模块嵌套代码风格偏好 - 2

    我的假设是moduleAmoduleBendend和moduleA::Bend是一样的。我能够从thisblog找到解决方案,thisSOthread和andthisSOthread.为什么以及什么时候应该更喜欢紧凑语法A::B而不是另一个,因为它显然有一个缺点?我有一种直觉,它可能与性能有关,因为在更多命名空间中查找常量需要更多计算。但是我无法通过对普通类进行基准测试来验证这一点。 最佳答案 这两种写作方法经常被混淆。首先要说的是,据我所知,没有可衡量的性能差异。(在下面的书面示例中不断查找)最明显的区别,可能也是最著名的,是你的

  7. ruby - 寻找通过阅读代码确定编程语言的ruby gem? - 2

    几个月前,我读了一篇关于ruby​​gem的博客文章,它可以通过阅读代码本身来确定编程语言。对于我的生活,我不记得博客或gem的名称。谷歌搜索“ruby编程语言猜测”及其变体也无济于事。有人碰巧知道相关gem的名称吗? 最佳答案 是这个吗:http://github.com/chrislo/sourceclassifier/tree/master 关于ruby-寻找通过阅读代码确定编程语言的rubygem?,我们在StackOverflow上找到一个类似的问题:

  8. ruby - Net::HTTP 获取源代码和状态 - 2

    我目前正在使用以下方法获取页面的源代码:Net::HTTP.get(URI.parse(page.url))我还想获取HTTP状态,而无需发出第二个请求。有没有办法用另一种方法做到这一点?我一直在查看文档,但似乎找不到我要找的东西。 最佳答案 在我看来,除非您需要一些真正的低级访问或控制,否则最好使用Ruby的内置Open::URI模块:require'open-uri'io=open('http://www.example.org/')#=>#body=io.read[0,50]#=>"["200","OK"]io.base_ur

  9. 程序员如何提高代码能力? - 2

    前言作为一名程序员,自己的本质工作就是做程序开发,那么程序开发的时候最直接的体现就是代码,检验一个程序员技术水平的一个核心环节就是开发时候的代码能力。众所周知,程序开发的水平提升是一个循序渐进的过程,每一位程序员都是从“菜鸟”变成“大神”的,所以程序员在程序开发过程中的代码能力也是根据平时开发中的业务实践来积累和提升的。提高代码能力核心要素程序员要想提高自身代码能力,尤其是新晋程序员的代码能力有很大的提升空间的时候,需要针对性的去提高自己的代码能力。提高代码能力其实有几个比较关键的点,只要把握住这些方面,就能很好的、快速的提高自己的一部分代码能力。1、多去阅读开源项目,如有机会可以亲自参与开源

  10. 7个大一C语言必学的程序 / C语言经典代码大全 - 2

    嗨~大家好,这里是可莉!今天给大家带来的是7个C语言的经典基础代码~那一起往下看下去把【程序一】打印100到200之间的素数#includeintmain(){ inti; for(i=100;i 【程序二】输出乘法口诀表#includeintmain(){inti;for(i=1;i 【程序三】判断1000年---2000年之间的闰年#includeintmain(){intyear;for(year=1000;year 【程序四】给定两个整形变量的值,将两个值的内容进行交换。这里提供两种方法来进行交换,第一种为创建临时变量来进行交换,第二种是不创建临时变量而直接进行交换。1.创建临时变量来

随机推荐