123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304 |
- -- Usage:
- -- # verbose
- -- nvim -l scripts/lintcommit.lua main --trace
- --
- -- # silent
- -- nvim -l scripts/lintcommit.lua main
- --
- -- # self-test
- -- nvim -l scripts/lintcommit.lua _test
- --- @type table<string,fun(opt: LintcommitOptions)>
- local M = {}
- local _trace = false
- -- Print message
- local function p(s)
- vim.cmd('set verbose=1')
- vim.api.nvim_echo({ { s, '' } }, false, {})
- vim.cmd('set verbose=0')
- end
- -- Executes and returns the output of `cmd`, or nil on failure.
- --
- -- Prints `cmd` if `trace` is enabled.
- local function run(cmd, or_die)
- if _trace then
- p('run: ' .. vim.inspect(cmd))
- end
- local rv = vim.trim(vim.fn.system(cmd)) or ''
- if vim.v.shell_error ~= 0 then
- if or_die then
- p(rv)
- os.exit(1)
- end
- return nil
- end
- return rv
- end
- -- Returns nil if the given commit message is valid, or returns a string
- -- message explaining why it is invalid.
- local function validate_commit(commit_message)
- local commit_split = vim.split(commit_message, ':', { plain = true })
- -- Return nil if the type is vim-patch since most of the normal rules don't
- -- apply.
- if commit_split[1] == 'vim-patch' then
- return nil
- end
- -- Check that message isn't too long.
- if commit_message:len() > 80 then
- return [[Commit message is too long, a maximum of 80 characters is allowed.]]
- end
- local before_colon = commit_split[1]
- local after_idx = 2
- if before_colon:match('^[^%(]*%([^%)]*$') then
- -- Need to find the end of commit scope when commit scope contains colons.
- while after_idx <= vim.tbl_count(commit_split) do
- after_idx = after_idx + 1
- if commit_split[after_idx - 1]:find(')') then
- break
- end
- end
- end
- if after_idx > vim.tbl_count(commit_split) then
- return [[Commit message does not include colons.]]
- end
- local after_colon_split = {}
- while after_idx <= vim.tbl_count(commit_split) do
- table.insert(after_colon_split, commit_split[after_idx])
- after_idx = after_idx + 1
- end
- local after_colon = table.concat(after_colon_split, ':')
- -- Check if commit introduces a breaking change.
- if vim.endswith(before_colon, '!') then
- before_colon = before_colon:sub(1, -2)
- end
- -- Check if type is correct
- local type = vim.split(before_colon, '(', { plain = true })[1]
- local allowed_types =
- { 'build', 'ci', 'docs', 'feat', 'fix', 'perf', 'refactor', 'revert', 'test', 'vim-patch' }
- if not vim.tbl_contains(allowed_types, type) then
- return string.format(
- [[Invalid commit type "%s". Allowed types are:
- %s.
- If none of these seem appropriate then use "fix"]],
- type,
- vim.inspect(allowed_types)
- )
- end
- -- Check if scope is appropriate
- if before_colon:match('%(') then
- local scope = vim.trim(commit_message:match('%((.-)%)'))
- if scope == '' then
- return [[Scope can't be empty]]
- end
- if vim.startswith(scope, 'nvim_') then
- return [[Scope should be "api" instead of "nvim_..."]]
- end
- local alternative_scope = {
- ['filetype.vim'] = 'filetype',
- ['filetype.lua'] = 'filetype',
- ['tree-sitter'] = 'treesitter',
- ['ts'] = 'treesitter',
- ['hl'] = 'highlight',
- }
- if alternative_scope[scope] then
- return ('Scope should be "%s" instead of "%s"'):format(alternative_scope[scope], scope)
- end
- end
- -- Check that description doesn't end with a period
- if vim.endswith(after_colon, '.') then
- return [[Description ends with a period (".").]]
- end
- -- Check that description starts with a whitespace.
- if after_colon:sub(1, 1) ~= ' ' then
- return [[There should be a whitespace after the colon.]]
- end
- -- Check that description doesn't start with multiple whitespaces.
- if after_colon:sub(1, 2) == ' ' then
- return [[There should only be one whitespace after the colon.]]
- end
- -- Allow lowercase or ALL_UPPER but not Titlecase.
- if after_colon:match('^ *%u%l') then
- return [[Description first word should not be Capitalized.]]
- end
- -- Check that description isn't just whitespaces
- if vim.trim(after_colon) == '' then
- return [[Description shouldn't be empty.]]
- end
- return nil
- end
- --- @param opt? LintcommitOptions
- function M.main(opt)
- _trace = not opt or not not opt.trace
- local branch = run({ 'git', 'rev-parse', '--abbrev-ref', 'HEAD' }, true)
- -- TODO(justinmk): check $GITHUB_REF
- local ancestor = run({ 'git', 'merge-base', 'origin/master', branch })
- if not ancestor then
- ancestor = run({ 'git', 'merge-base', 'upstream/master', branch })
- end
- local commits_str = run({ 'git', 'rev-list', ancestor .. '..' .. branch }, true)
- assert(commits_str)
- local commits = {} --- @type string[]
- for substring in commits_str:gmatch('%S+') do
- table.insert(commits, substring)
- end
- local failed = 0
- for _, commit_id in ipairs(commits) do
- local msg = run({ 'git', 'show', '-s', '--format=%s', commit_id })
- if vim.v.shell_error ~= 0 then
- p('Invalid commit-id: ' .. commit_id .. '"')
- else
- local invalid_msg = validate_commit(msg)
- if invalid_msg then
- failed = failed + 1
- -- Some breathing room
- if failed == 1 then
- p('\n')
- end
- p(string.format(
- [[
- Invalid commit message: "%s"
- Commit: %s
- %s
- ]],
- msg,
- commit_id,
- invalid_msg
- ))
- end
- end
- end
- if failed > 0 then
- p([[
- See also:
- https://github.com/neovim/neovim/blob/master/CONTRIBUTING.md#commit-messages
- ]])
- os.exit(1)
- else
- p('')
- end
- end
- function M._test()
- -- message:expected_result
- local test_cases = {
- ['ci: normal message'] = true,
- ['build: normal message'] = true,
- ['docs: normal message'] = true,
- ['feat: normal message'] = true,
- ['fix: normal message'] = true,
- ['perf: normal message'] = true,
- ['refactor: normal message'] = true,
- ['revert: normal message'] = true,
- ['test: normal message'] = true,
- ['ci(window): message with scope'] = true,
- ['ci!: message with breaking change'] = true,
- ['ci(tui)!: message with scope and breaking change'] = true,
- ['vim-patch:8.2.3374: Pyret files are not recognized (#15642)'] = true,
- ['vim-patch:8.1.1195,8.2.{3417,3419}'] = true,
- ['revert: "ci: use continue-on-error instead of "|| true""'] = true,
- ['fixup'] = false,
- ['fixup: commit message'] = false,
- ['fixup! commit message'] = false,
- [':no type before colon 1'] = false,
- [' :no type before colon 2'] = false,
- [' :no type before colon 3'] = false,
- ['ci(empty description):'] = false,
- ['ci(only whitespace as description): '] = false,
- ['docs(multiple whitespaces as description): '] = false,
- ['revert(multiple whitespaces and then characters as description): description'] = false,
- ['ci no colon after type'] = false,
- ['test: extra space after colon'] = false,
- ['ci: tab after colon'] = false,
- ['ci:no space after colon'] = false,
- ['ci :extra space before colon'] = false,
- ['refactor(): empty scope'] = false,
- ['ci( ): whitespace as scope'] = false,
- ['ci: period at end of sentence.'] = false,
- ['ci: period: at end of sentence.'] = false,
- ['ci: Capitalized first word'] = false,
- ['ci: UPPER_CASE First Word'] = true,
- ['unknown: using unknown type'] = false,
- ['feat: foo:bar'] = true,
- ['feat: :foo:bar'] = true,
- ['feat: :Foo:Bar'] = true,
- ['feat(something): foo:bar'] = true,
- ['feat(something): :foo:bar'] = true,
- ['feat(something): :Foo:Bar'] = true,
- ['feat(:grep): read from pipe'] = true,
- ['feat(:grep/:make): read from pipe'] = true,
- ['feat(:grep): foo:bar'] = true,
- ['feat(:grep/:make): foo:bar'] = true,
- ['feat(:grep)'] = false,
- ['feat(:grep/:make)'] = false,
- ['feat(:grep'] = false,
- ['feat(:grep/:make'] = false,
- ["ci: you're saying this commit message just goes on and on and on and on and on and on for way too long?"] = false,
- }
- local failed = 0
- for message, expected in pairs(test_cases) do
- local is_valid = (nil == validate_commit(message))
- if is_valid ~= expected then
- failed = failed + 1
- p(
- string.format('[ FAIL ]: expected=%s, got=%s\n input: "%s"', expected, is_valid, message)
- )
- end
- end
- if failed > 0 then
- os.exit(1)
- end
- end
- --- @class LintcommitOptions
- --- @field trace? boolean
- local opt = {}
- for _, a in ipairs(arg) do
- if vim.startswith(a, '--') then
- local nm, val = a:sub(3), true
- if vim.startswith(a, '--no') then
- nm, val = a:sub(5), false
- end
- if nm == 'trace' then
- opt.trace = val
- end
- end
- end
- for _, a in ipairs(arg) do
- if M[a] then
- M[a](opt)
- end
- end
|