summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorsumneko <sumneko@hotmail.com>2021-11-28 01:51:15 +0800
committersumneko <sumneko@hotmail.com>2021-11-28 01:51:15 +0800
commit6b7bbbd97cd8166d0371ec235771ff14ce5e3cf8 (patch)
tree45579463fb1bc470d47fd220d60901f52f3e27d9
parentef96be3ff9a7d51e4087c6d492c0c1606ad9f273 (diff)
downloadlua-language-server-6b7bbbd97cd8166d0371ec235771ff14ce5e3cf8.zip
update
-rw-r--r--script/workspace/loading.lua77
-rw-r--r--script/workspace/workspace.lua137
2 files changed, 128 insertions, 86 deletions
diff --git a/script/workspace/loading.lua b/script/workspace/loading.lua
new file mode 100644
index 00000000..1142b943
--- /dev/null
+++ b/script/workspace/loading.lua
@@ -0,0 +1,77 @@
+local progress = require 'progress'
+local lang = require 'language'
+local await = require 'await'
+
+---@class workspace.loading
+---@field _bar progress
+---@field _stash function[]
+local mt = {}
+mt.__index = mt
+
+mt._loadLock = false
+mt.read = 0
+mt.max = 0
+mt.preload = 0
+
+function mt:update()
+ self._bar:setMessage(('%d/%d'):format(self.read, self.max))
+ self._bar:setPercentage(self.read / self.max * 100.0)
+end
+
+function mt:load()
+ self:update()
+ if self._loadLock then
+ return
+ end
+ self._loadLock = true
+ ---@async
+ await.call(function ()
+ while true do
+ local loader = table.remove(self._stash)
+ if not loader then
+ break
+ end
+ loader()
+ await.delay()
+ end
+ self._loadLock = false
+ end)
+end
+
+---@async
+function mt:loadAll(callback)
+ while true do
+ callback()
+ self:update()
+ if self.read >= self.max then
+ break
+ end
+ await.sleep(0.1)
+ end
+end
+
+function mt:isFinished()
+ return self.read >= self.max
+end
+
+function mt:remove()
+ self._bar:remove()
+end
+
+function mt:__close()
+ self:remove()
+end
+
+---@class workspace.loading.manager
+local m = {}
+
+---@return workspace.loading
+function m.create(scp)
+ local loading = setmetatable({
+ _bar = progress.create(lang.script('WORKSPACE_LOADING', scp.uri)),
+ _stash = {},
+ }, mt)
+ return loading
+end
+
+return m
diff --git a/script/workspace/workspace.lua b/script/workspace/workspace.lua
index aee61366..a40f2a3d 100644
--- a/script/workspace/workspace.lua
+++ b/script/workspace/workspace.lua
@@ -70,24 +70,21 @@ local globInteferFace = {
--- 创建排除文件匹配器
---@async
-function m.getNativeMatcher()
- if not m.rootPath then
- return nil
- end
- if m.nativeMatcher then
- return m.nativeMatcher
+function m.getNativeMatcher(scp)
+ if scp:get 'nativeMatcher' then
+ return scp:get 'nativeMatcher'
end
local pattern = {}
-- config.get(nil, 'files.exclude'
- for path, ignore in pairs(config.get(nil, 'files.exclude')) do
+ for path, ignore in pairs(config.get(scp.uri, 'files.exclude')) do
if ignore then
log.info('Ignore by exclude:', path)
pattern[#pattern+1] = path
end
end
-- config.get(nil, 'workspace.useGitIgnore'
- if config.get(nil, 'Lua.workspace.useGitIgnore') then
+ if config.get(scp.uri, 'Lua.workspace.useGitIgnore') then
local buf = pub.awaitTask('loadFile', furi.encode(m.rootPath .. '/.gitignore'))
if buf then
for line in buf:gmatch '[^\r\n]+' do
@@ -108,7 +105,7 @@ function m.getNativeMatcher()
end
end
-- config.get(nil, 'workspace.ignoreSubmodules'
- if config.get(nil, 'Lua.workspace.ignoreSubmodules') then
+ if config.get(scp.uri, 'Lua.workspace.ignoreSubmodules') then
local buf = pub.awaitTask('loadFile', furi.encode(m.rootPath .. '/.gitmodules'))
if buf then
for path in buf:gmatch('path = ([^\r\n]+)') do
@@ -118,7 +115,7 @@ function m.getNativeMatcher()
end
end
-- config.get(nil, 'workspace.library'
- for path in pairs(config.get(nil, 'Lua.workspace.library')) do
+ for path in pairs(config.get(scp.uri, 'Lua.workspace.library')) do
path = m.getAbsolutePath(path)
if path then
log.info('Ignore by library:', path)
@@ -126,50 +123,57 @@ function m.getNativeMatcher()
end
end
-- config.get(nil, 'workspace.ignoreDir'
- for _, path in ipairs(config.get(nil, 'Lua.workspace.ignoreDir')) do
+ for _, path in ipairs(config.get(scp.uri, 'Lua.workspace.ignoreDir')) do
log.info('Ignore directory:', path)
pattern[#pattern+1] = path
end
- m.nativeMatcher = glob.gitignore(pattern, m.matchOption, globInteferFace)
- m.nativeMatcher:setOption('root', m.rootPath)
+ local matcher = glob.gitignore(pattern, {
+ root = furi.decode(scp.uri),
+ ignoreCase = platform.OS == 'Windows',
+ }, globInteferFace)
- m.nativeVersion = config.get(nil, 'version')
- return m.nativeMatcher
+ scp:set('nativeMatcher', matcher)
+ return matcher
end
--- 创建代码库筛选器
-function m.getLibraryMatchers()
- if m.libraryMatchers then
- return m.libraryMatchers
+---@param scp scope
+function m.getLibraryMatchers(scp)
+ if scp:get 'nativeMatcher' then
+ return scp:get 'nativeMatcher'
end
local librarys = {}
- for path in pairs(config.get(nil, 'Lua.workspace.library')) do
+ for path in pairs(config.get(scp.uri, 'Lua.workspace.library')) do
path = m.getAbsolutePath(path)
if path then
librarys[m.normalize(path)] = true
end
end
+ -- TODO
if library.metaPath then
librarys[m.normalize(library.metaPath)] = true
end
- m.libraryMatchers = {}
+
+ local matchers = {}
for path in pairs(librarys) do
if fs.exists(fs.path(path)) then
local nPath = fs.absolute(fs.path(path)):string()
- local matcher = glob.gitignore(true, m.matchOption, globInteferFace)
- matcher:setOption('root', path)
- log.debug('getLibraryMatchers', path, nPath)
- m.libraryMatchers[#m.libraryMatchers+1] = {
- path = nPath,
+ local matcher = glob.gitignore(true, {
+ root = path,
+ ignoreCase = platform.OS == 'Windows',
+ }, globInteferFace)
+ matchers[#matchers+1] = {
+ uri = furi.encode(nPath),
matcher = matcher
}
end
end
- m.libraryVersion = config.get(nil, 'version')
- return m.libraryMatchers
+ scp:set('nativeMatcher', matchers)
+
+ return matchers
end
--- 文件是否被忽略
@@ -195,7 +199,7 @@ function m.isValidLuaUri(uri)
return true
end
-local function loadFileFactory(root, progressData, isLibrary)
+local function loadFileFactory(scp, progressData, isLibrary)
return function (path) ---@async
local uri = furi.encode(path)
if files.isLua(uri) then
@@ -305,80 +309,41 @@ function m.awaitPreload(scp)
await.close 'preload'
await.setID 'preload'
await.sleep(0.1)
- diagnostic.pause(scp)
- if scp:get 'watchers' then
- for _, dispose in ipairs(scp:get 'watchers') do
+ local watchers = scp:get 'watchers'
+ if watchers then
+ for _, dispose in ipairs(watchers) do
dispose()
end
end
- scp:set('watchers', {})
+ watchers = {}
+ scp:set('watchers', watchers)
- local ld = loading.create(scp)
+ local ld <close> = loading.create(scp)
- local progressBar <close> = progress.create(lang.script('WORKSPACE_LOADING', scp.uri))
- local progressData = {
- max = 0,
- read = 0,
- preload = 0,
- loaders = {},
- update = function (self)
- progressBar:setMessage(('%d/%d'):format(self.read, self.max))
- progressBar:setPercentage(self.read / self.max * 100)
- m.fileLoaded = self.read
- m.fileFound = self.max
- end
- }
- log.info('Preload start.')
+ log.info('Preload start:', scp.uri)
- local nativeLoader = loadFileFactory(m.rootPath, progressData)
- local native = m.getNativeMatcher()
- local librarys = m.getLibraryMatchers()
+ local nativeLoader = loadFileFactory(scp, ld)
+ local native = m.getNativeMatcher(scp)
+ local librarys = m.getLibraryMatchers(scp)
if native then
log.info('Scan files at:', m.rootPath)
native:scan(m.rootPath, nativeLoader)
end
- for _, library in ipairs(librarys) do
- local libraryLoader = loadFileFactory(library.path, progressData, true)
- log.info('Scan library at:', library.path)
- library.matcher:scan(library.path, libraryLoader)
- m.watchers[#m.watchers+1] = fw.watch(library.path)
- end
-
- local isLoadingFiles = false
- local function loadSomeFiles()
- if isLoadingFiles then
- return
- end
- await.call(function () ---@async
- isLoadingFiles = true
- while true do
- local loader = table.remove(progressData.loaders)
- if not loader then
- break
- end
- loader()
- await.delay()
- end
- isLoadingFiles = false
- end)
+ for _, libMatcher in ipairs(librarys) do
+ local libraryLoader = loadFileFactory(scp, ld, libMatcher.uri)
+ log.info('Scan library at:', libMatcher.uri)
+ libMatcher.matcher:scan(furi.decode(libMatcher.uri), libraryLoader)
+ watchers[#watchers+1] = fw.watch(furi.decode(libMatcher.uri))
end
- log.info(('Found %d files.'):format(progressData.max))
- while true do
- loadSomeFiles()
- log.info(('Loaded %d/%d files'):format(progressData.read, progressData.max))
- progressData:update()
- if progressData.read >= progressData.max then
- break
- end
+ log.info(('Found %d files.'):format(ld.max))
+ while not ld:isFinished() do
+ log.info(('Loaded %d/%d files'):format(ld.read, ld.max))
+ ld:load()
await.sleep(0.1)
end
- progressBar:remove()
-
log.info('Preload finish.')
-
- diagnostic.start()
end
--- 查找符合指定file path的所有uri