mirror of
https://github.com/neovim/neovim.git
synced 2024-12-19 10:45:16 -07:00
304 lines
8.5 KiB
Lua
304 lines
8.5 KiB
Lua
-- 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
|