diff options
-rw-r--r-- | server/main.lua | 1 | ||||
-rw-r--r-- | server/src/core/document_symbol.lua | 288 | ||||
-rw-r--r-- | server/src/core/hover/hover.lua | 6 | ||||
-rw-r--r-- | server/src/core/hover/name.lua | 69 | ||||
-rw-r--r-- | server/src/core/name.lua | 70 | ||||
-rw-r--r-- | server/src/method/textDocument/documentSymbol.lua | 28 | ||||
-rw-r--r-- | server/src/service.lua | 4 | ||||
-rw-r--r-- | server/src/vm/local.lua | 9 | ||||
-rw-r--r-- | server/src/vm/vm.lua | 4 | ||||
-rw-r--r-- | server/test/crossfile/document_symbol.lua | 4 | ||||
-rw-r--r-- | server/test/document_symbol/init.lua | 33 |
11 files changed, 289 insertions, 227 deletions
diff --git a/server/main.lua b/server/main.lua index 4f3dcf71..dd0c53d3 100644 --- a/server/main.lua +++ b/server/main.lua @@ -7,6 +7,7 @@ package.path = (ROOT / 'src' / '?.lua'):string() .. ';' .. (ROOT / 'src' / '?' / 'init.lua'):string() --collectgarbage('generational') +collectgarbage("setpause", 100) log = require 'log' log.init(ROOT, ROOT / 'log' / 'service.log') diff --git a/server/src/core/document_symbol.lua b/server/src/core/document_symbol.lua index e22fd4db..ac81277c 100644 --- a/server/src/core/document_symbol.lua +++ b/server/src/core/document_symbol.lua @@ -1,5 +1,5 @@ local hoverFunction = require 'core.hover.function' -local hoverName = require 'core.hover.name' +local getName = require 'core.name' local hover = require 'core.hover' local SymbolKind = { @@ -31,171 +31,183 @@ local SymbolKind = { TypeParameter = 26, } -local function isFirstSet(source, value) - if source:action() ~= 'set' then - return false +local function buildLocal(vm, source, used, callback) + local vars = source[1] + local exps = source[2] + if vars.type ~= 'list' then + vars = {vars} end - local firstSrc = value:eachInfo(function (info, src) - if info.type == 'set' then - return src - end - end) - if not firstSrc then - return false + if not exps or exps.type ~= 'list' then + exps = {exps} end - if firstSrc ~= source then - return false + for i, var in ipairs(vars) do + local exp = exps[i] + local data = {} + local loc = var:bindLocal() + data.name = loc:getName() + data.range = { var.start, var.finish } + data.selectionRange = { var.start, var.finish } + if exp then + local hvr = hover(var) + if exp.type == 'function' then + data.kind = SymbolKind.Function + else + data.kind = SymbolKind.Variable + end + data.detail = hvr.label:gsub('[\r\n]', '') + data.valueRange = { exp.start, exp.finish } + used[exp] = true + else + data.kind = SymbolKind.Variable + data.detail = '' + data.valueRange = { var.start, var.finish } + end + callback(data) end - return true end -local function buildLocal(vm, source, callback) - local loc = source:bindLocal() - local value = loc:getInitValue() - local hvr = hover(source) - if not hvr then - return +local function buildSet(vm, source, used, callback) + local vars = source[1] + local exps = source[2] + if vars.type ~= 'list' then + vars = {vars} end - local kind - if value:getType() == 'function' then - kind = SymbolKind.Function - elseif source:get 'table index' then - kind = SymbolKind.Class - else - kind = SymbolKind.Variable - end - local valueSource = value:getSource() - if not valueSource or valueSource.start == 0 or value.uri ~= vm.uri then - valueSource = source - end - local name = hvr.name - if name == '' then - name = tostring(source[1] or '') - end - -- 由于范围不允许交叉,为了支持 local x, y, z = 1, 2, 3 的形式 - -- 范围只能限定在变量上 - -- 而 local function xx() 的形式范围会包含整个 function - if source.start > valueSource.start and source.finish < valueSource.finish then - callback { - name = name, - detail = hvr.label:gsub('[\r\n]', ''), - kind = kind, - range = { valueSource.start, valueSource.finish }, - selectionRange = { source.start, source.finish }, - valueRange = { valueSource.start, valueSource.finish }, - } - else - callback { - name = name, - detail = hvr.label:gsub('[\r\n]', ''), - kind = kind, - range = { source.start, source.finish }, - selectionRange = { source.start, source.finish }, - valueRange = { valueSource.start, valueSource.finish }, - } + if not exps or exps.type ~= 'list' then + exps = {exps} + end + for i, var in ipairs(vars) do + if var:bindLocal() then + goto CONTINUE + end + local exp = exps[i] + local data = {} + data.name = getName(var) + data.range = { var.start, var.finish } + data.selectionRange = { var.start, var.finish } + if exp then + local hvr = hover(var) + if not hvr then + goto CONTINUE + end + if exp.type == 'function' then + data.kind = SymbolKind.Function + else + data.kind = SymbolKind.Property + end + data.detail = hvr.label:gsub('[\r\n]', '') + data.valueRange = { exp.start, exp.finish } + used[exp] = true + else + data.kind = SymbolKind.Property + data.detail = '' + data.valueRange = { var.start, var.finish } + end + callback(data) + :: CONTINUE :: end end -local function buildSet(vm, source, callback) - if source:bindLocal() then - return - end - local value = source:bindValue() - local hvr = hover(source) - if not hvr then - return - end - local kind - if value:getFunction() then - local func = value:getFunction() - if func:getObject() then - kind = SymbolKind.Field +local function buildPair(vm, source, used, callback) + local var = source[1] + local exp = source[2] + local data = {} + data.name = getName(var) + data.range = { var.start, var.finish } + data.selectionRange = { var.start, var.finish } + if exp then + local hvr = hover(var) + if not hvr then + return + end + if exp.type == 'function' then + data.kind = SymbolKind.Function else - kind = SymbolKind.Function + data.kind = SymbolKind.Class end - elseif source:get 'table index' then - kind = SymbolKind.Class + data.detail = hvr.label:gsub('[\r\n]', '') + data.valueRange = { exp.start, exp.finish } + used[exp] = true else - kind = SymbolKind.Property - end - local valueSource = value:getSource() - if not valueSource or valueSource.start == 0 or value.uri ~= vm.uri then - valueSource = source - end - local name = hvr.name - if name == '' then - name = tostring(source[1] or '') - end - -- 由于范围不允许交叉,为了支持 x, y, z = 1, 2, 3 的形式 - -- 范围只能限定在变量上 - -- 而 function xx() 的形式范围会包含整个 function - if source.start > valueSource.start and source.finish < valueSource.finish then - callback { - name = name, - -- 前端不支持多行 - detail = hvr.label:gsub('[\r\n]', ''), - kind = kind, - range = { valueSource.start, valueSource.finish }, - selectionRange = { source.start, source.finish }, - valueRange = { valueSource.start, valueSource.finish }, - } - elseif isFirstSet(source, value) then - callback { - name = name, - -- 前端不支持多行 - detail = hvr.label:gsub('[\r\n]', ''), - kind = kind, - range = { source.start, source.finish }, - selectionRange = { source.start, source.finish }, - valueRange = { valueSource.start, valueSource.finish }, - } - else - callback { - name = name, - -- 前端不支持多行 - detail = hvr.label:gsub('[\r\n]', ''), - kind = kind, - range = { source.start, source.finish }, - selectionRange = { source.start, source.finish }, - valueRange = { source.start, source.finish }, - } + data.kind = SymbolKind.Class + data.detail = '' + data.valueRange = { var.start, var.finish } end + callback(data) end -local function buildReturn(vm, source, callback) +local function buildLocalFunction(vm, source, used, callback) local value = source:bindFunction() if not value then return end - local hvr = hoverFunction('', value:getFunction()) + local name = getName(source.name) + local hvr = hoverFunction(name, value:getFunction()) if not hvr then return end local kind = SymbolKind.Function callback { - name = '', - -- 前端不支持多行 + name = name, detail = hvr.label:gsub('[\r\n]', ''), kind = kind, range = { source.start, source.finish }, - selectionRange = { source.start, source.start }, + selectionRange = { source.name.start, source.name.finish }, valueRange = { source.start, source.finish }, } end -local function buildSource(vm, source, callback) - if source:action() == 'local' then - buildLocal(vm, source, callback) + +local function buildFunction(vm, source, used, callback) + if used[source] then return end - if source:action() == 'set' then - buildSet(vm, source, callback) + local value = source:bindFunction() + if not value then return end - if source.type == 'return' then - for _, src in ipairs(source) do - buildReturn(vm, src, callback) - end + local name = getName(source.name) + local func = value:getFunction() + local hvr = hoverFunction(name, func, func:getObject()) + if not hvr then + return + end + local data = {} + data.name = name + data.detail = hvr.label:gsub('[\r\n]', '') + data.range = { source.start, source.finish } + data.valueRange = { source.start, source.finish } + if source.name then + data.selectionRange = { source.name.start, source.name.finish } + else + data.selectionRange = { source.start, source.start } + end + if func:getObject() then + data.kind = SymbolKind.Field + else + data.kind = SymbolKind.Function + end + callback(data) +end + +local function buildSource(vm, source, used, callback) + if source.type == 'local' then + buildLocal(vm, source, used, callback) + return + end + if source.type == 'set' then + buildSet(vm, source, used, callback) + return + end + if source.type == 'pair' then + buildPair(vm, source, used, callback) + return + end + if source.type == 'localfunction' then + buildLocalFunction(vm, source, used, callback) + return + end + if source.type == 'function' then + buildFunction(vm, source, used, callback) return end end @@ -212,13 +224,10 @@ local function packChild(symbols, finish, kind) end symbols[#symbols] = nil symbol.children = packChild(symbols, symbol.valueRange[2], symbol.kind) - if symbol.kind == SymbolKind.Class and kind == SymbolKind.Function then - else - if not t then - t = {} - end - t[#t+1] = symbol + if not t then + t = {} end + t[#t+1] = symbol end return t end @@ -226,7 +235,7 @@ end local function packSymbols(symbols) -- 按照start位置反向排序 table.sort(symbols, function (a, b) - return a.valueRange[1] > b.valueRange[1] + return a.range[1] > b.range[1] end) -- 处理嵌套 return packChild(symbols, math.maxinteger, SymbolKind.Function) @@ -234,9 +243,10 @@ end return function (vm) local symbols = {} + local used = {} for _, source in ipairs(vm.sources) do - buildSource(vm, source, function (data) + buildSource(vm, source, used, function (data) symbols[#symbols+1] = data end) end diff --git a/server/src/core/hover/hover.lua b/server/src/core/hover/hover.lua index c7aba17e..f09bfce4 100644 --- a/server/src/core/hover/hover.lua +++ b/server/src/core/hover/hover.lua @@ -220,5 +220,11 @@ return function (source, lsp, select) if source.type == 'name' and source:bindValue() then return hoverAsValue(source, lsp, select) end + if source.type == 'simple' then + source = source[#source] + if source.type == 'name' and source:bindValue() then + return hoverAsValue(source, lsp, select) + end + end return nil end diff --git a/server/src/core/hover/name.lua b/server/src/core/hover/name.lua index 52bcfef4..e5880884 100644 --- a/server/src/core/hover/name.lua +++ b/server/src/core/hover/name.lua @@ -1,4 +1,9 @@ +local getName = require 'core.name' + return function (source) + if not source then + return '' + end local value = source:bindValue() if not value then return '' @@ -22,67 +27,5 @@ return function (source) return name or '' end - local key - if declarat:get 'simple' then - local simple = declarat:get 'simple' - local chars = {} - for i, obj in ipairs(simple) do - if obj.type == 'name' then - chars[i] = obj[1] - elseif obj.type == 'index' then - chars[i] = '[?]' - elseif obj.type == 'call' then - chars[i] = '(?)' - elseif obj.type == ':' then - chars[i] = ':' - elseif obj.type == '.' then - chars[i] = '.' - else - chars[i] = '*' .. obj.type - end - if obj == declarat then - break - end - end - key = table.concat(chars) - elseif declarat.type == 'name' then - key = declarat[1] - elseif declarat.type == 'string' then - key = ('%q'):format(declarat[1]) - elseif declarat.type == 'number' or declarat.type == 'boolean' then - key = tostring(declarat[1]) - elseif declarat.type == 'simple' then - local chars = {} - for i, obj in ipairs(declarat) do - if obj.type == 'name' then - chars[i] = obj[1] - elseif obj.type == 'index' then - chars[i] = '[?]' - elseif obj.type == 'call' then - chars[i] = '(?)' - elseif obj.type == ':' then - chars[i] = ':' - elseif obj.type == '.' then - chars[i] = '.' - else - chars[i] = '*' .. obj.type - end - end - -- 这里有个特殊处理 - -- function mt:func() 以 mt.func 的形式调用时 - -- hover 显示为 mt.func(self) - if chars[#chars-1] == ':' then - if not source:get 'object' then - chars[#chars-1] = '.' - end - elseif chars[#chars-1] == '.' then - if source:get 'object' then - chars[#chars-1] = ':' - end - end - key = table.concat(chars) - else - key = '' - end - return key + return getName(declarat, source) end diff --git a/server/src/core/name.lua b/server/src/core/name.lua new file mode 100644 index 00000000..54947974 --- /dev/null +++ b/server/src/core/name.lua @@ -0,0 +1,70 @@ +return function (source, caller) + if not source then + return '' + end + local key + if source:get 'simple' then + local simple = source:get 'simple' + local chars = {} + for i, obj in ipairs(simple) do + if obj.type == 'name' then + chars[i] = obj[1] + elseif obj.type == 'index' then + chars[i] = '[?]' + elseif obj.type == 'call' then + chars[i] = '(?)' + elseif obj.type == ':' then + chars[i] = ':' + elseif obj.type == '.' then + chars[i] = '.' + else + chars[i] = '*' .. obj.type + end + if obj == source then + break + end + end + key = table.concat(chars) + elseif source.type == 'name' then + key = source[1] + elseif source.type == 'string' then + key = ('%q'):format(source[1]) + elseif source.type == 'number' or source.type == 'boolean' then + key = tostring(source[1]) + elseif source.type == 'simple' then + local chars = {} + for i, obj in ipairs(source) do + if obj.type == 'name' then + chars[i] = obj[1] + elseif obj.type == 'index' then + chars[i] = '[?]' + elseif obj.type == 'call' then + chars[i] = '(?)' + elseif obj.type == ':' then + chars[i] = ':' + elseif obj.type == '.' then + chars[i] = '.' + else + chars[i] = '*' .. obj.type + end + end + -- 这里有个特殊处理 + -- function mt:func() 以 mt.func 的形式调用时 + -- hover 显示为 mt.func(self) + if caller then + if chars[#chars-1] == ':' then + if not caller:get 'object' then + chars[#chars-1] = '.' + end + elseif chars[#chars-1] == '.' then + if caller:get 'object' then + chars[#chars-1] = ':' + end + end + end + key = table.concat(chars) + else + key = '' + end + return key +end diff --git a/server/src/method/textDocument/documentSymbol.lua b/server/src/method/textDocument/documentSymbol.lua index 2f9a6b8a..9b21131d 100644 --- a/server/src/method/textDocument/documentSymbol.lua +++ b/server/src/method/textDocument/documentSymbol.lua @@ -1,6 +1,8 @@ local core = require 'core' local lang = require 'language' +local timerCache = {} + local function posToRange(lines, start, finish) local start_row, start_col = lines:rowcol(start) local finish_row, finish_col = lines:rowcol(finish) @@ -32,19 +34,29 @@ end return function (lsp, params) local uri = params.textDocument.uri + + if timerCache[uri] then + timerCache[uri]:remove() + timerCache[uri] = nil + end + local vm, lines = lsp:loadVM(uri) if not vm then return nil end - local symbols = core.documentSymbol(vm) - if not symbols then - return nil - end + return function (response) + timerCache[uri] = ac.wait(0.5, function () + local symbols = core.documentSymbol(vm) + if not symbols then + return nil + end - for _, symbol in ipairs(symbols) do - convertRange(lines, symbol) - end + for _, symbol in ipairs(symbols) do + convertRange(lines, symbol) + end - return symbols + response(symbols) + end) + end end diff --git a/server/src/service.lua b/server/src/service.lua index 0edfe935..0ec563e8 100644 --- a/server/src/service.lua +++ b/server/src/service.lua @@ -91,9 +91,9 @@ function mt:_doProto(proto) return end if type(response) == 'function' then - return function (final) + response(function (final) self:responseProto(id, final) - end + end) else self:responseProto(id, response, err) end diff --git a/server/src/vm/local.lua b/server/src/vm/local.lua index 8f9ee3e4..cada88af 100644 --- a/server/src/vm/local.lua +++ b/server/src/vm/local.lua @@ -21,14 +21,6 @@ function mt:getValue() return self.value end -function mt:setInitValue(value) - self.initValue = value -end - -function mt:getInitValue() - return self.initValue -end - function mt:addInfo(tp, source) if not source then error('No source') @@ -157,7 +149,6 @@ return function (name, source, value) name = name, source = id, value = value, - initValue = value, _info = {}, }, mt) return self diff --git a/server/src/vm/vm.lua b/server/src/vm/vm.lua index 093122e4..9fcd57ae 100644 --- a/server/src/vm/vm.lua +++ b/server/src/vm/vm.lua @@ -40,6 +40,7 @@ function mt:buildTable(source) if obj.type == 'pair' then local value = self:getFirstInMulti(self:getExp(obj[2])) local key = obj[1] + self:instantSource(obj) self:instantSource(key) key:bindValue(value, 'set') if key.index then @@ -794,6 +795,7 @@ function mt:doSet(action) if not action[2] then return end + self:instantSource(action) -- 要先计算值 local vars = action[1] local exps = action[2] @@ -814,6 +816,7 @@ function mt:doSet(action) end function mt:doLocal(action) + self:instantSource(action) local vars = action[1] local exps = action[2] local values @@ -966,7 +969,6 @@ function mt:doLocalFunction(action) local func = self:buildFunction(action) func:addInfo('local', name) loc:setValue(func) - loc:setInitValue(func) end end end diff --git a/server/test/crossfile/document_symbol.lua b/server/test/crossfile/document_symbol.lua index 1ce571d9..015e3638 100644 --- a/server/test/crossfile/document_symbol.lua +++ b/server/test/crossfile/document_symbol.lua @@ -109,10 +109,10 @@ local t = { [1] = { name = 'x', detail = EXISTS, - kind = SymbolKind.Function, + kind = SymbolKind.Class, range = {17, 17}, selectionRange = {17, 17}, - valueRange = {17, 17}, + valueRange = {21, 31}, }, } } diff --git a/server/test/document_symbol/init.lua b/server/test/document_symbol/init.lua index e17acd9f..929c2253 100644 --- a/server/test/document_symbol/init.lua +++ b/server/test/document_symbol/init.lua @@ -170,7 +170,7 @@ end detail = 'function mt:add()', kind = SymbolKind.Field, range = {1, 21}, - selectionRange = {13, 15}, + selectionRange = {10, 15}, valueRange = {1, 21}, } } @@ -285,7 +285,7 @@ local z }, [4] = { name = 'z', - detail = 'local z: any', + detail = '', kind = SymbolKind.Variable, range = {102, 102}, selectionRange = {102, 102}, @@ -411,12 +411,39 @@ end kind = SymbolKind.Variable, range = {27, 27}, selectionRange = {27, 27}, - valueRange = {27, 27}, + valueRange = {31, 33}, } } }, } +TEST [[ +local t = { + a = 1, + b = 2, +} + +local v = t +]]{ + [1] = { + name = 't', + detail = EXISTS, + kind = SymbolKind.Variable, + range = {7, 7}, + selectionRange = {7, 7}, + valueRange = {11, 35}, + children = EXISTS, + }, + [2] = { + name = 'v', + detail = EXISTS, + kind = SymbolKind.Variable, + range = {44, 44}, + selectionRange = {44, 44}, + valueRange = {48, 48}, + }, +} + -- 临时 local fs = require 'bee.filesystem' local function testIfExit(path) |