summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--server/main.lua1
-rw-r--r--server/src/core/document_symbol.lua288
-rw-r--r--server/src/core/hover/hover.lua6
-rw-r--r--server/src/core/hover/name.lua69
-rw-r--r--server/src/core/name.lua70
-rw-r--r--server/src/method/textDocument/documentSymbol.lua28
-rw-r--r--server/src/service.lua4
-rw-r--r--server/src/vm/local.lua9
-rw-r--r--server/src/vm/vm.lua4
-rw-r--r--server/test/crossfile/document_symbol.lua4
-rw-r--r--server/test/document_symbol/init.lua33
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)