131 lines
3.2 KiB
Lua
131 lines
3.2 KiB
Lua
|
local files = require 'files'
|
||
|
local guide = require 'parser.guide'
|
||
|
local define = require 'proto.define'
|
||
|
local lang = require 'language'
|
||
|
local vm = require 'vm.vm'
|
||
|
local config = require 'config.config'
|
||
|
local glob = require 'glob'
|
||
|
|
||
|
local function hasGet(loc)
|
||
|
if not loc.ref then
|
||
|
return false
|
||
|
end
|
||
|
local weak
|
||
|
for _, ref in ipairs(loc.ref) do
|
||
|
if ref.type == 'getlocal' then
|
||
|
if not ref.next then
|
||
|
return 'strong'
|
||
|
end
|
||
|
local nextType = ref.next.type
|
||
|
if nextType ~= 'setmethod'
|
||
|
and nextType ~= 'setfield'
|
||
|
and nextType ~= 'setindex' then
|
||
|
return 'strong'
|
||
|
else
|
||
|
weak = true
|
||
|
end
|
||
|
end
|
||
|
end
|
||
|
if weak then
|
||
|
return 'weak'
|
||
|
else
|
||
|
return nil
|
||
|
end
|
||
|
end
|
||
|
|
||
|
local function isMyTable(loc)
|
||
|
local value = loc.value
|
||
|
if value and value.type == 'table' then
|
||
|
return true
|
||
|
end
|
||
|
return false
|
||
|
end
|
||
|
|
||
|
local function isToBeClosed(source)
|
||
|
if not source.attrs then
|
||
|
return false
|
||
|
end
|
||
|
for _, attr in ipairs(source.attrs) do
|
||
|
if attr[1] == 'close' then
|
||
|
return true
|
||
|
end
|
||
|
end
|
||
|
return false
|
||
|
end
|
||
|
|
||
|
local function isDocClass(source)
|
||
|
if not source.bindDocs then
|
||
|
return false
|
||
|
end
|
||
|
for _, doc in ipairs(source.bindDocs) do
|
||
|
if doc.type == 'doc.class' then
|
||
|
return true
|
||
|
end
|
||
|
end
|
||
|
return false
|
||
|
end
|
||
|
|
||
|
---@param source parser.object
|
||
|
local function isDeclareFunctionParam(source)
|
||
|
if source.parent.type ~= 'funcargs' then
|
||
|
return false
|
||
|
end
|
||
|
local func = source.parent.parent
|
||
|
return vm.isEmptyFunction(func)
|
||
|
end
|
||
|
|
||
|
return function (uri, callback)
|
||
|
local ast = files.getState(uri)
|
||
|
if not ast then
|
||
|
return
|
||
|
end
|
||
|
|
||
|
local isMeta = vm.isMetaFile(uri)
|
||
|
local ignorePatterns = config.get(uri, 'Lua.diagnostics.unusedLocalExclude')
|
||
|
local ignore = glob.glob(ignorePatterns)
|
||
|
guide.eachSourceType(ast.ast, 'local', function (source)
|
||
|
local name = source[1]
|
||
|
if name == '_'
|
||
|
or name == ast.ENVMode then
|
||
|
return
|
||
|
end
|
||
|
if ignore(name) then
|
||
|
return
|
||
|
end
|
||
|
if isToBeClosed(source) then
|
||
|
return
|
||
|
end
|
||
|
if isDocClass(source) then
|
||
|
return
|
||
|
end
|
||
|
if isMeta and isDeclareFunctionParam(source) then
|
||
|
return
|
||
|
end
|
||
|
local data = hasGet(source)
|
||
|
if data == 'strong' then
|
||
|
return
|
||
|
end
|
||
|
if data == 'weak' then
|
||
|
if not isMyTable(source) then
|
||
|
return
|
||
|
end
|
||
|
end
|
||
|
callback {
|
||
|
start = source.start,
|
||
|
finish = source.finish,
|
||
|
tags = { define.DiagnosticTag.Unnecessary },
|
||
|
message = lang.script('DIAG_UNUSED_LOCAL', name),
|
||
|
}
|
||
|
if source.ref then
|
||
|
for _, ref in ipairs(source.ref) do
|
||
|
callback {
|
||
|
start = ref.start,
|
||
|
finish = ref.finish,
|
||
|
tags = { define.DiagnosticTag.Unnecessary },
|
||
|
message = lang.script('DIAG_UNUSED_LOCAL', name),
|
||
|
}
|
||
|
end
|
||
|
end
|
||
|
end)
|
||
|
end
|