commitlint 是一个提交验证工具。原理是可以在实际的 git commit 提交到远程仓库之前使用 git 钩子来验证信息。提交不符合规则的信息将会被阻止提交到远程仓库。

    • @commitlint/cli:对commit的消息进行格式检查。
    • @commitlint/config-conventional:通用的提交规范,对应还有很多其他配置,比如angular的提交规范@commitlint/config-angular

    安装 commitlint检查工具 以及 conventional 规范包

    1. pnpm add @commitlint/cli @commitlint/config-conventional -D

    安装后添加配置文件.commitlintrc.js 或者 commitlint.config.js

    1. const fs = require('fs')
    2. const path = require('path')
    3. const { execSync } = require('child_process')
    4. const scopes = fs
    5. .readdirSync(path.resolve(__dirname, 'src'), { withFileTypes: true })
    6. .filter((dirent) => dirent.isDirectory())
    7. .map((dirent) => dirent.name.replace(/s$/, ''))
    8. // precomputed scope
    9. const scopeComplete = execSync('git status --porcelain || true')
    10. .toString()
    11. .trim()
    12. .split('\n')
    13. .find((r) => ~r.indexOf('M src'))
    14. ?.replace(/(\/)/g, '%%')
    15. ?.match(/src%%((\w|-)*)/)?.[1]
    16. ?.replace(/s$/, '')
    17. /** @type {import('cz-git').UserConfig} */
    18. module.exports = {
    19. ignores: [(commit) => commit.includes('init')],
    20. // 继承的规则
    21. extends: ['@commitlint/config-conventional'],
    22. // 定义规则类型
    23. rules: {
    24. 'body-leading-blank': [2, 'always'],
    25. 'footer-leading-blank': [1, 'always'],
    26. 'header-max-length': [2, 'always', 108],
    27. 'subject-empty': [2, 'never'],
    28. 'type-empty': [2, 'never'],
    29. // subject 大小写不做校验
    30. 'subject-case': [0],
    31. // type 类型定义,表示 git 提交的 type 必须在以下类型范围内
    32. 'type-enum': [
    33. 2,
    34. 'always',
    35. [
    36. 'feat',
    37. 'fix',
    38. 'perf',
    39. 'style',
    40. 'docs',
    41. 'test',
    42. 'refactor',
    43. 'build',
    44. 'ci',
    45. 'chore',
    46. 'revert',
    47. 'wip',
    48. 'workflow',
    49. 'types',
    50. 'release'
    51. ]
    52. ]
    53. },
    54. prompt: {
    55. /** @use `yarn commit :f` */
    56. alias: {
    57. f: 'docs: fix typos',
    58. r: 'docs: update README',
    59. s: 'style: update code format',
    60. b: 'build: bump dependencies',
    61. c: 'chore: update config'
    62. },
    63. customScopesAlign: !scopeComplete ? 'top' : 'bottom',
    64. defaultScope: scopeComplete,
    65. scopes: [...scopes, 'mock'],
    66. allowEmptyIssuePrefixs: false,
    67. allowCustomIssuePrefixs: false,
    68. // English
    69. typesAppend: [
    70. { value: 'wip', name: 'wip: work in process' },
    71. { value: 'workflow', name: 'workflow: workflow improvements' },
    72. { value: 'types', name: 'types: type definition file changes' }
    73. ]
    74. // 中英文对照版
    75. // messages: {
    76. // type: '选择你要提交的类型 :',
    77. // scope: '选择一个提交范围 (可选):',
    78. // customScope: '请输入自定义的提交范围 :',
    79. // subject: '填写简短精炼的变更描述 :\n',
    80. // body: '填写更加详细的变更描述 (可选)。使用 "|" 换行 :\n',
    81. // breaking: '列举非兼容性重大的变更 (可选)。使用 "|" 换行 :\n',
    82. // footerPrefixsSelect: '选择关联issue前缀 (可选):',
    83. // customFooterPrefixs: '输入自定义issue前缀 :',
    84. // footer: '列举关联issue (可选) 例如: #31, #I3244 :\n',
    85. // confirmCommit: '是否提交或修改commit ?',
    86. // },
    87. // types: [
    88. // { value: 'feat', name: 'feat: 新增功能' },
    89. // { value: 'fix', name: 'fix: 修复缺陷' },
    90. // { value: 'docs', name: 'docs: 文档变更' },
    91. // { value: 'style', name: 'style: 代码格式' },
    92. // { value: 'refactor', name: 'refactor: 代码重构' },
    93. // { value: 'perf', name: 'perf: 性能优化' },
    94. // { value: 'test', name: 'test: 添加疏漏测试或已有测试改动' },
    95. // { value: 'build', name: 'build: 构建流程、外部依赖变更 (如升级 npm 包、修改打包配置等)' },
    96. // { value: 'ci', name: 'ci: 修改 CI 配置、脚本' },
    97. // { value: 'revert', name: 'revert: 回滚 commit' },
    98. // { value: 'chore', name: 'chore: 对构建过程或辅助工具和库的更改 (不影响源文件、测试用例)' },
    99. // { value: 'wip', name: 'wip: 正在开发中' },
    100. // { value: 'workflow', name: 'workflow: 工作流程改进' },
    101. // { value: 'types', name: 'types: 类型定义文件修改' },
    102. // ],
    103. // emptyScopesAlias: 'empty: 不填写',
    104. // customScopesAlias: 'custom: 自定义',
    105. }
    106. }

    为了可以在每次 commit 时执行 commitlint 来 检查我们输入的 message,我们还需要用到 husky。

    使用 husky add 添加钩子,.husky文件夹下生成文件 commit-msg

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

    “$1” 在win上的终端有时候不能正确的识别,也可以直接手动添加 commit-msg 文件

    1. #!/usr/bin/env sh
    2. . "$(dirname -- "$0")/_/husky.sh"
    3. npx --no-install commitlint --edit $1

    测试一下,commit 时不加 type 会提示对应错误。
    image.png