voidrice/.local/share/nvim/mason/packages/lua-language-server/libexec/script/filewatch.lua
2024-01-27 17:26:04 +01:00

128 lines
3.1 KiB
Lua

local fw = require 'bee.filewatch'
local fs = require 'bee.filesystem'
local plat = require 'bee.platform'
local await = require 'await'
local files = require 'files'
local MODIFY = 1 << 0
local RENAME = 1 << 1
local function isExists(filename)
local path = fs.path(filename)
local suc, exists = pcall(fs.exists, path)
if not suc or not exists then
return false
end
if plat.OS ~= 'Windows' then
return true
end
local suc, res = pcall(fs.fullpath, path)
if not suc then
return false
end
if res :string():gsub('^%w+:', string.lower)
~= path:string():gsub('^%w+:', string.lower) then
return false
end
return true
end
---@class filewatch
local m = {}
m._eventList = {}
m._watchings = {}
---@async
---@param path string
---@param recursive boolean
---@param filter? fun(path: string):boolean
function m.watch(path, recursive, filter)
if path == '' or not fs.is_directory(fs.path(path)) then
return function () end
end
if m._watchings[path] then
m._watchings[path].count = m._watchings[path].count + 1
else
local watch = fw.create()
if recursive then
watch:set_recursive(true)
watch:set_follow_symlinks(true)
watch:set_filter(filter)
end
log.debug('Watch add:', path)
watch:add(path)
m._watchings[path] = {
count = 1,
watch = watch,
}
log.debug('fw.add', path)
end
local removed
return function ()
if removed then
return
end
removed = true
m._watchings[path].count = m._watchings[path].count - 1
if m._watchings[path].count == 0 then
m._watchings[path] = nil
log.debug('fw.remove', path)
end
end
end
---@param callback async fun(ev: string, path: string)
function m.event(callback)
m._eventList[#m._eventList+1] = callback
end
function m._callEvent(ev, path)
for _, callback in ipairs(m._eventList) do
await.call(function ()
callback(ev, path)
end)
end
end
function m.update()
local collect
for _, watching in pairs(m._watchings) do
local watch = watching.watch
for _ = 1, 10000 do
local ev, path = watch:select()
if not ev then
break
end
path = files.normalize(path)
log.debug('filewatch:', ev, path)
if not collect then
collect = {}
end
if ev == 'modify' then
collect[path] = (collect[path] or 0) | MODIFY
elseif ev == 'rename' then
collect[path] = (collect[path] or 0) | RENAME
end
end
end
if not collect or not next(collect) then
return
end
for path, flag in pairs(collect) do
if flag & RENAME ~= 0 then
if isExists(path) then
m._callEvent('create', path)
else
m._callEvent('delete', path)
end
elseif flag & MODIFY ~= 0 then
m._callEvent('change', path)
end
end
end
return m