feat(filetype): fall back to file extension when matching from hashbang (#22140)

If nothing matched in match_from_hashbang, also check the file extension table.
For a hashbang like '#!/bin/env foo', this will set the filetype to 'fooscript'
assuming the filetype for the 'foo' extension is 'fooscript' in the extension
table.
This commit is contained in:
Jonas Strittmatter 2023-02-14 00:04:16 +01:00 committed by GitHub
parent 1d6bb0892b
commit b518aceaa8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 24 additions and 5 deletions

View File

@ -2604,7 +2604,9 @@ function M.match(args)
-- If the function tries to use the filename that is nil then it will fail, -- If the function tries to use the filename that is nil then it will fail,
-- but this enables checks which do not need a filename to still work. -- but this enables checks which do not need a filename to still work.
local ok local ok
ok, ft = pcall(require('vim.filetype.detect').match_contents, contents, name) ok, ft = pcall(require('vim.filetype.detect').match_contents, contents, name, function(ext)
return dispatch(extension[ext], name, bufnr)
end)
if ok and ft then if ok and ft then
return ft return ft
end end

View File

@ -1420,7 +1420,7 @@ local patterns_hashbang = {
---@private ---@private
-- File starts with "#!". -- File starts with "#!".
local function match_from_hashbang(contents, path) local function match_from_hashbang(contents, path, dispatch_extension)
local first_line = contents[1] local first_line = contents[1]
-- Check for a line like "#!/usr/bin/env {options} bash". Turn it into -- Check for a line like "#!/usr/bin/env {options} bash". Turn it into
-- "#!/usr/bin/bash" to make matching easier. -- "#!/usr/bin/bash" to make matching easier.
@ -1473,6 +1473,11 @@ local function match_from_hashbang(contents, path)
return ft return ft
end end
end end
-- If nothing matched, check the extension table. For a hashbang like
-- '#!/bin/env foo', this will set the filetype to 'fooscript' assuming
-- the filetype for the 'foo' extension is 'fooscript' in the extension table.
return dispatch_extension(name)
end end
local patterns_text = { local patterns_text = {
@ -1652,10 +1657,10 @@ local function match_from_text(contents, path)
return cvs_diff(path, contents) return cvs_diff(path, contents)
end end
M.match_contents = function(contents, path) function M.match_contents(contents, path, dispatch_extension)
local first_line = contents[1] local first_line = contents[1]
if first_line:find('^#!') then if first_line:find('^#!') then
return match_from_hashbang(contents, path) return match_from_hashbang(contents, path, dispatch_extension)
else else
return match_from_text(contents, path) return match_from_text(contents, path)
end end

View File

@ -98,10 +98,22 @@ describe('vim.filetype', function()
it('works with contents #22180', function() it('works with contents #22180', function()
eq('sh', exec_lua [[ eq('sh', exec_lua [[
-- Needs to be set so detect#sh doesn't fail -- Needs to be set so detect#sh doesn't fail
vim.g.ft_ignore_pat = "\\.\\(Z\\|gz\\|bz2\\|zip\\|tgz\\)$" vim.g.ft_ignore_pat = '\\.\\(Z\\|gz\\|bz2\\|zip\\|tgz\\)$'
return vim.filetype.match({ contents = { '#!/usr/bin/env bash' } }) return vim.filetype.match({ contents = { '#!/usr/bin/env bash' } })
]]) ]])
end) end)
it('considers extension mappings when matching from hashbang', function()
eq('fooscript', exec_lua [[
vim.filetype.add({
extension = {
foo = 'fooscript',
}
})
return vim.filetype.match({ contents = { '#!/usr/bin/env foo' } })
]])
end)
end) end)
describe('filetype.lua', function() describe('filetype.lua', function()