51工具盒子

依楼听风雨
笑看云卷云舒,淡观潮起潮落

Git lint 相关

前言 {#前言}

集成 lint 代码规范工具 中,已经集成了对项目代码的检查,但这并不是强制的,可能没有在编辑器中配置 ESLint,或者忽视了命令行中的错误提示,导致错误的代码被提交到仓库中。

我们需要在 Git 提交代码前,通过 Git Hooks 强制对代码进行增量检查 (每次都进行全量 lint 检查是非常耗时的)。并配置 commitlint 对提交信息也进行检查。

示例仓库:qxchuckle/monorepo-test

参考:
【从 0 到 1 搭建 Vue 组件库框架】3. 集成 lint 代码规范工具
一文带你彻底学会 Git Hooks 配置
[Git Hooks] 在代码提交前自动格式化代码
(前端工程化配置) 使用commitlint校验git commit message
commit规范+commitlint+CHANGELOG自动生成一条龙服务
【前端】代码Git提交规范之约定式提交和Commitizen简化提交流程

commitlint {#commitlint}

commitlint 用于规范化 Git 提交信息。

安装 {#安装}

|-----------|--------------------------------------------------------------------| | 1 | pnpm i -wD @commitlint/config-conventional @commitlint/cli |

在根目录创建 commitlint.config.mjs,继承默认的 @commitlint/config-conventional 规范集。(这些个 lint 工具的配置文件长得都大差不差)。

安装 @archoleat/commitlint-define-config 以获得配置提示。 commitlint.config.mjs

|-------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 | // @ts-check import { defineConfig } from '@archoleat/commitlint-define-config'; export default defineConfig({ extends: ['@commitlint/config-conventional'], }); |

@commitlint/config-conventional 规定了 Conventional Commits 式的提交信息,详见 Git 规范与实践

gitmoji {#gitmoji}

如果你使用了 gitmoji,可以继承 commitlint-config-gitmoji 配置。

Git Hooks {#Git-Hooks}

commitlint 无法单独使用。因为提交信息发生在 git commit 阶段,而 git commit 时,控制台已经被占用,我们无法输入其他命令。

但 Git 提供了 Git Hooks 功能,它能在特定的重要动作发生时触发自定义脚本。
可以在 commit 动作发生时执行 commitlint 脚本,通过 commit-msg 钩子判断所提交的信息是否符合规范。

Hook 就是在执行某个事件之前或之后进行一些其他额外的操作。

Git Hooks 的实现非常简单,就是就 .git/hooks 文件下,保存了一些对应阶段的 shell 脚本文件 ,其中默认存在的 .sample 文件是示例文件,将该后缀去掉,如 pre-commit 文件,就是钩子文件了,Git 会在特定阶段自动执行对应的脚本文件

Husky {#Husky}

我们当然可以手动创建这些钩子文件,但是 .git 文件夹并不会被提交到仓库中,其他人在 clone 项目时,无法使用这些钩子。

这就需要使用 Husky 来管理、配置钩子了。

|-------------|-----------------------------------------------------| | 1 2 | pnpm i -wD husky npx husky init # 初始化 husky |

这会在项目根目录创建 .husky 文件夹,里面存放了一些和 Git Hooks 同名的钩子文件。并将 core.hooksPath 配置为该文件夹。初始化完成时只有 pre-commit 文件。

其原理很清晰明了,操作钩子时,只需在 .husky 文件夹中操作对应的文件即可。当然可以使用系统自带的 echo 等命令完成,看个人喜好。

|-----------|---------------------------------------------| | 1 | echo "npm test" > .husky/pre-commit |

此外,初始化时还会修改 package.json 文件,添加 scripts.prepare 脚本,以在 npm 安装依赖前执行 husky 命令,用于正确配置 core.hooksPath.husky 文件夹。 package.json

|-------------------|-----------------------------------------------| | 1 2 3 4 5 | { "scripts": { "prepare": "husky" } } |

配置 commit-msg 钩子 {#配置-commit-msg-钩子}

添加 .husky/commit-msg 文件,调用 commitlint 进行检查。 .husky/commit-msg

|-----------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 | #!/usr/bin/env sh # --no 不按照任何缺失的包,确保 npx 只运行已经安装的包。 # --:这是一个分隔符,用于区分 npx 的选项和要运行的命令的选项。所有在 -- 之后的内容都会被视为要运行的命令的参数。 # HUSKY_GIT_PARAMS 被移除。取而代之的是 Git 参数应该直接在脚本中使用(例如 $1)。 # --edit 选项用于指定要检查的提交消息文件。$1 是传递给脚本的第一个参数,通常是提交消息文件的路径。 npx --no -- commitlint --edit $1 |

现在,不规范的提交将被拒绝。

|---------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 | > git commit -m "测试不规范提交" ⧗ input: 测试不规范提交 ✖ 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 script failed (code 1) > git commit -m "test: 测试规范提交" [main 1bd745a] test: 测试规范提交 2 files changed, 7 insertions(+), 2 deletions(-) create mode 100644 .husky/commit-msg delete mode 100644 .husky/pre-commit |

lint-staged 实现增量检查 {#lint-staged-实现增量检查}

目前的 ESLintStylelint 都是全量检查,这是耗时且不必要的,因为往往只有修改后即将提交的文件才需要检查。

很容易想到的是配合 Git 实现增量检查,即只检查 staged(暂存区) 的文件。lint-staged 就是这样一个工具。

安装 {#安装-1}

|-----------|--------------------------------| | 1 | pnpm i -wD lint-staged |

lint-staged 支持通过 glob 模式匹配,对暂存区的文件列表进行分类过滤,以实现对不同的文件应用不同检查的效果。

创建 lint-staged.config.mjs 配置文件。 lint-staged.config.mjs

|------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 | export default { // 对于 js、ts 脚本文件,应用 eslint '**/*.{js,jsx,tsx,ts}': ['eslint --fix'], // 对于 css scss 文件,应用 stylelint '**/*.{scss,css}': ['stylelint --fix'], // Vue 文件由于同时包含模板、样式、脚本,因此 eslint、stylelint 都要使用 '**/*.vue': ['eslint --fix', 'stylelint --fix'], // 用 prettier 修复所有文件的格式 '**/*': ['prettier --write'], }; |

配置 pre-commit 钩子 {#配置-pre-commit-钩子}

lint-staged 要处理的是暂存区文件,所以需要使用 Git 的 pre-commit 钩子,实现在 commit 发生之前对发生变化的文件进行 Lint 扫描,若 Lint 抛出错误,说明此次准备提交的文件存在代码规范的问题,提交失败。

这需要我们再次用到 husky。 .husky/pre-commit

|-------------|---------------------------------------------------| | 1 2 | #!/usr/bin/env sh npx --no -- lint-staged |

现在提交代码时,会检查并尝试修复暂存区的文件。

commitizen {#commitizen}

配合 VSCode 可以使用 Conventional Commitsgit-commit-plugin 等插件快速生成符合 Angular 团队规范的提交信息。

而命令行工具更加通用,commitizen 提供 cz 命令替代 commit 命令,它会引导开发者通过一系列问题来填写提交信息,确保提交信息符合规范。

安装 {#安装-2}

|-------------|--------------------------------------------------------------------------------------------------------------| | 1 2 | npm i -g commitizen pnpm i -wD commitizen # 推荐在项目中安装,全局模式下, 需要 ~/.czrc 配置文件, 为 commitizen 指定 Adapter |

还需要安装额外的适配器,否则 repo 就不是对 Commitizen 友好的,git cz 的工作方式会与 git commit 完全相同。

cz-conventional-changelog 是 commitizen 的首选适配器。

|-----------|----------------------------------------------| | 1 | pnpm i -wD cz-conventional-changelog |

编辑 package.json 文件,添加 config.commitizen 字段,指定适配器。 package.json

|-------------------------|-----------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 | "scripts": { "commit": "cz" } "config": { "commitizen": { "path": "cz-conventional-changelog" } } |

现在,使用 pnpm run commit 命令提交代码,会引导你填写符合规范的提交信息。

|------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 | > pnpm run commit cz-cli@4.3.0, cz-conventional-changelog@3.3.0 ? Select the type of change that you're committing: (Use arrow keys) > feat: A new feature fix: A bug fix docs: Documentation only changes style: Changes that do not affect the meaning of the code (white-space, formatting, missing semi-colons, etc) refactor: A code change that neither fixes a bug nor adds a feature perf: A code change that improves performance test: Adding missing tests or correcting existi |

更换适配器 {#更换适配器}

cz-conventional-changelog 适配器提示信息都是英文的,不能自定义配置,我们使用更灵活的 cz-customizable 适配器。

|-----------|------------------------------------| | 1 | pnpm i -wD cz-customizable |

更换适配器: package.json

|-------------------|--------------------------------------------------------------------| | 1 2 3 4 5 | "config": { "commitizen": { "path": "cz-customizable" } }, |

在 root 下新建 .cz-config.js 配置文件,配置适配器。配置选项文档

吐槽,.cz-config.js 只能是 CJS 模块,因为 cz-customizable 用的是 require 引入。 .cz-config.js

|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 | module.exports = { // 定义提交类型 types: [ { value: 'feat', name: 'feat: 新功能', }, { value: 'fix', name: 'fix: 修复bug', }, { value: 'init', name: 'init: 初始化', }, { value: ':pencil2: docs', name: 'docs: 文档变更', }, { value: 'style', name: 'style: 代码的样式美化', }, { value: 'refactor', name: 'refactor: 重构', }, { value: 'perf', name: 'perf: 性能优化', }, { value: 'test', name: 'test: 测试', }, { value: 'revert', name: 'revert: 回退', }, { value: 'build', name: 'build: 打包', }, { value: 'chore', name: 'chore: 构建/工程依赖/工具', }, { value: 'ci', name: 'ci: CI related changes', }, ], // 自定义提示信息 messages: { type: '请选择提交类型(必填)', customScope: '请输入文件修改范围(可选)', subject: '请简要描述提交(必填)', body: '请输入详细描述(可选)', breaking: '列出任何BREAKING CHANGES(可选)', footer: '请输入要关闭的issue(可选)', confirmCommit: '确定提交此说明吗?', }, // 允许自定义 scopes 范围 allowCustomScopes: true, // 当提交类型为feat、fix时才有破坏性修改选项 allowBreakingChanges: ['feat', 'fix'], // 简短描述长度限制 subjectLimit: 72, }; |

再次运行,现在可以看到自定义的提示信息。cz -> lint -> commit,一套流程行云流水。

|------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| | 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 | > pnpm run commit > pnpm-workspace-test@ commit C:\chuckle\qx\study_demo\Monorepo\pnpmWorkspaceTest > cz cz-cli@4.3.0, cz-customizable@7.2.1 All lines except first will be wrapped after 100 characters. ? 请选择提交类型(必填) feat: 新功能 ? 请简要描述提交(必填) 集成 commitizen ? 请输入详细描述(可选) ? 列出任何BREAKING CHANGES(可选) ? 请输入要关闭的issue(可选) ###--------------------------------------------------------### feat(custom): 集成 commitizen ###--------------------------------------------------------### ? 确定提交此说明吗? Yes [STARTED] Preparing lint-staged... [COMPLETED] Preparing lint-staged... [STARTED] Running tasks for staged files... [STARTED] lint-staged.config.mjs --- 3 files [STARTED] **/*.{js,jsx,tsx,ts} --- 1 file [STARTED] **/*.{scss,css} --- 0 files [STARTED] **/*.vue --- 0 files [STARTED] **/* --- 3 files [SKIPPED] **/*.{scss,css} --- no files [SKIPPED] **/*.vue --- no files [STARTED] eslint --fix [STARTED] prettier --write [COMPLETED] prettier --write [COMPLETED] **/* --- 3 files [COMPLETED] eslint --fix [COMPLETED] **/*.{js,jsx,tsx,ts} --- 1 file [COMPLETED] lint-staged.config.mjs --- 3 files [COMPLETED] Running tasks for staged files... [STARTED] Applying modifications from tasks... [COMPLETED] Applying modifications from tasks... [STARTED] Cleaning up temporary files... [COMPLETED] Cleaning up temporary files... [main c0c24d9] feat(custom): 集成 commitizen 3 files changed, 815 insertions(+) create mode 100644 .cz-config.js |

CHANGELOG 自动生成 {#CHANGELOG-自动生成}

CHANGELOG 用于记录项目所有的 commit 信息并归类版本。

可以使用 VSCode 插件 whatchangedcommit-and-tag-versionconventional-changelog 自动生成 CHANGELOG。

赞(1)
未经允许不得转载:工具盒子 » Git lint 相关