diff options
author | Tom Lau <tomandfatboy@gmail.com> | 2024-08-23 09:54:24 +0800 |
---|---|---|
committer | Tom Lau <tomandfatboy@gmail.com> | 2024-08-24 21:13:24 +0800 |
commit | e09d181224da52fd6f82d228a8bcdffba58564d8 (patch) | |
tree | 3b7859718559c85bdaf5ad1b1e2f543fc3e12fdb | |
parent | ba8f90eb0fab18ce8aee2bdbf7007dc63050381d (diff) | |
download | lua-language-server-e09d181224da52fd6f82d228a8bcdffba58564d8.zip |
fix: improve function type narrow by checking params' literal identical
-rw-r--r-- | changelog.md | 1 | ||||
-rw-r--r-- | script/vm/function.lua | 59 | ||||
-rw-r--r-- | script/vm/value.lua | 7 |
3 files changed, 54 insertions, 13 deletions
diff --git a/changelog.md b/changelog.md index 2f0ea8f7..2a35ff3d 100644 --- a/changelog.md +++ b/changelog.md @@ -2,6 +2,7 @@ ## Unreleased <!-- Add all new changes here. They will be moved under a version at release --> +* `FIX` Improve type narrow by checking exact match on literal type params ## 3.10.5 `2024-8-19` diff --git a/script/vm/function.lua b/script/vm/function.lua index 1e308317..7a15ac5a 100644 --- a/script/vm/function.lua +++ b/script/vm/function.lua @@ -353,6 +353,35 @@ local function isAllParamMatched(uri, args, params) return true end +---@param uri uri +---@param args parser.object[] +---@param func parser.object +---@return number +local function calcFunctionMatchScore(uri, args, func) + if vm.isVarargFunctionWithOverloads(func) + or not isAllParamMatched(uri, args, func.args) + then + return -1 + end + local matchScore = 0 + for i = 1, math.min(#args, #func.args) do + local arg, param = args[i], func.args[i] + local defLiterals, literalsCount = vm.getLiterals(param) + if defLiterals then + for n in vm.compileNode(arg):eachObject() do + -- if param's literals map contains arg's literal, this is narrower than a subtype match + if defLiterals[guide.getLiteral(n)] then + -- the more the literals defined in the param, the less bonus score will be added + -- this favors matching overload param with exact literal value, over alias/enum that has many literal values + matchScore = matchScore + 1/literalsCount + break + end + end + end + end + return matchScore +end + ---@param func parser.object ---@param args? parser.object[] ---@return parser.object[]? @@ -365,21 +394,29 @@ function vm.getExactMatchedFunctions(func, args) return funcs end local uri = guide.getUri(func) - local needRemove + local matchScores = {} for i, n in ipairs(funcs) do - if vm.isVarargFunctionWithOverloads(n) - or not isAllParamMatched(uri, args, n.args) then - if not needRemove then - needRemove = {} - end - needRemove[#needRemove+1] = i - end + matchScores[i] = calcFunctionMatchScore(uri, args, n) + end + + local maxMatchScore = math.max(table.unpack(matchScores)) + if maxMatchScore == -1 then + -- all should be removed + return nil end - if not needRemove then + + local minMatchScore = math.min(table.unpack(matchScores)) + if minMatchScore == maxMatchScore then + -- all should be kept return funcs end - if #needRemove == #funcs then - return nil + + -- remove functions that have matchScore < maxMatchScore + local needRemove = {} + for i, matchScore in ipairs(matchScores) do + if matchScore < maxMatchScore then + needRemove[#needRemove + 1] = i + end end util.tableMultiRemove(funcs, needRemove) return funcs diff --git a/script/vm/value.lua b/script/vm/value.lua index 7eab4a8e..ce031357 100644 --- a/script/vm/value.lua +++ b/script/vm/value.lua @@ -213,11 +213,13 @@ end ---@param v vm.object ---@return table<any, boolean>? +---@return integer function vm.getLiterals(v) if not v then - return nil + return nil, 0 end local map + local count = 0 local node = vm.compileNode(v) for n in node:eachObject() do local literal @@ -237,7 +239,8 @@ function vm.getLiterals(v) map = {} end map[literal] = true + count = count + 1 end end - return map + return map, count end |