nomsu/files.lua

270 lines
6.6 KiB
Lua
Raw Normal View History

local lpeg = require('lpeg')
local re = require('re')
local Files = { }
assert(package.nomsupath, "No package.nomsupath was found")
local run_cmd
run_cmd = function(cmd)
local f = io.popen(cmd)
local lines
do
local _accum_0 = { }
local _len_0 = 1
for line in f:lines() do
_accum_0[_len_0] = line
_len_0 = _len_0 + 1
end
lines = _accum_0
end
if not (f:close()) then
return nil
end
return lines
end
local _SPOOFED_FILES = { }
local _FILE_CACHE = setmetatable({ }, {
__index = _SPOOFED_FILES
})
local _BROWSE_CACHE = { }
local _anon_number = 0
Files.spoof = function(filename, contents)
if not contents then
filename, contents = "<anonymous file #" .. tostring(_anon_number) .. ">", filename
_anon_number = _anon_number + 1
end
_SPOOFED_FILES[filename] = contents
return filename
end
Files.read = function(filename)
do
local file_contents = _FILE_CACHE[filename]
if file_contents then
return file_contents
end
end
if filename == 'stdin' then
local contents = io.read('*a')
Files.spoof('stdin', contents)
return contents
end
local file = io.open(filename)
if not (file) then
return nil
end
local contents = file:read("*a")
file:close()
_FILE_CACHE[filename] = contents
return contents
end
2018-06-23 17:24:28 -07:00
local match, gsub
do
local _obj_0 = string
2018-06-23 17:24:28 -07:00
match, gsub = _obj_0.match, _obj_0.gsub
end
2018-07-17 16:13:35 -07:00
local sanitize
sanitize = function(path)
path = gsub(path, "\\", "\\\\")
path = gsub(path, "`", "")
path = gsub(path, '"', '\\"')
path = gsub(path, "$", "")
return path
end
Files.exists = function(path)
if _SPOOFED_FILES[path] then
return true
end
2018-07-24 17:36:45 -07:00
if path == 'stdin' then
return true
end
if run_cmd("ls " .. tostring(sanitize(path))) then
2018-07-17 16:13:35 -07:00
return true
end
for nomsupath in package.nomsupath:gmatch("[^;]+") do
if run_cmd("ls " .. tostring(nomsupath) .. "/" .. tostring(sanitize(path))) then
return true
2018-07-17 16:13:35 -07:00
end
end
2018-07-17 16:13:35 -07:00
return false
end
local browse
browse = function(path)
if not (_BROWSE_CACHE[path]) then
local files
if _SPOOFED_FILES[path] then
_BROWSE_CACHE[path] = {
path
}
else
_BROWSE_CACHE[path] = run_cmd('find -L "' .. path .. '" -not -path "*/\\.*" -type f') or false
2018-07-17 16:13:35 -07:00
end
end
return _BROWSE_CACHE[path]
end
local ok, lfs = pcall(require, "lfs")
if ok then
2018-07-09 16:58:46 -07:00
local raw_file_exists
raw_file_exists = function(filename)
local mode = lfs.attributes(filename, 'mode')
2018-07-09 19:22:40 -07:00
if mode == 'file' or mode == 'directory' or mode == 'link' then
2018-07-09 16:58:46 -07:00
return true
else
return false
end
2018-07-09 16:58:46 -07:00
end
Files.exists = function(path)
if _SPOOFED_FILES[path] then
return true
end
2018-07-09 16:58:46 -07:00
if path == 'stdin' or raw_file_exists(path) then
return true
end
for nomsupath in package.nomsupath:gmatch("[^;]+") do
if raw_file_exists(nomsupath .. "/" .. path) then
return true
2018-07-09 16:58:46 -07:00
end
end
return false
end
browse = function(filename)
if not (_BROWSE_CACHE[filename]) then
2018-07-24 17:36:45 -07:00
if _SPOOFED_FILES[filename] or filename == 'stdin' then
_BROWSE_CACHE[filename] = {
2018-07-17 16:13:35 -07:00
filename
}
else
local file_type, err = lfs.attributes(filename, 'mode')
local _exp_0 = file_type
if "file" == _exp_0 or "char device" == _exp_0 then
_BROWSE_CACHE[filename] = {
filename
}
elseif "directory" == _exp_0 or "link" == _exp_0 then
local files = { }
for subfile in lfs.dir(filename) do
local _continue_0 = false
repeat
if subfile == "." or subfile == ".." then
_continue_0 = true
break
end
local _list_0 = (browse(filename .. "/" .. subfile) or { })
for _index_0 = 1, #_list_0 do
local f = _list_0[_index_0]
files[#files + 1] = f
end
_continue_0 = true
until true
if not _continue_0 then
break
2018-07-17 16:13:35 -07:00
end
end
_BROWSE_CACHE[filename] = files
else
_BROWSE_CACHE[filename] = false
end
end
2018-07-17 16:13:35 -07:00
end
return _BROWSE_CACHE[filename]
end
else
if not (run_cmd('find . -maxdepth 0')) then
local url
if jit then
url = 'https://github.com/spacewander/luafilesystem'
else
url = 'https://github.com/keplerproject/luafilesystem'
end
error("Could not find 'luafilesystem' module and couldn't run system command `find` (this might happen on Windows). Please install `luafilesystem` (which can be found at: " .. tostring(url) .. " or `luarocks install luafilesystem`)", 0)
end
2018-07-17 16:13:35 -07:00
end
Files.walk = function(path, flush_cache)
2018-07-17 16:13:35 -07:00
if flush_cache == nil then
flush_cache = false
end
if flush_cache then
_BROWSE_CACHE = { }
2018-07-09 16:58:46 -07:00
end
local files
if path == 'stdin' or _SPOOFED_FILES[path] then
2018-07-24 17:36:45 -07:00
files = {
path
}
elseif path:match("^[~/]") or path:match("^%./") or path:match("^%.%./") then
files = browse(path)
2018-07-24 17:36:45 -07:00
else
for nomsupath in package.nomsupath:gmatch("[^;]+") do
do
files = browse(nomsupath .. "/" .. path)
if files then
break
end
2018-07-09 16:58:46 -07:00
end
end
end
files = files or { }
do
local _accum_0 = { }
local _len_0 = 1
for _index_0 = 1, #files do
local f = files[_index_0]
_accum_0[_len_0] = gsub(f, "^%./", "")
_len_0 = _len_0 + 1
2018-07-17 16:13:35 -07:00
end
files = _accum_0
end
return ipairs(files)
end
local line_counter = re.compile([[ lines <- {| line (%nl line)* |}
line <- {} (!%nl .)*
]], {
nl = lpeg.P("\r") ^ -1 * lpeg.P("\n")
})
local _LINE_STARTS = { }
Files.get_line_starts = function(str)
if type(str) ~= 'string' then
str = tostring(str)
end
do
local starts = _LINE_STARTS[str]
if starts then
return starts
end
end
local line_starts = line_counter:match(str)
_LINE_STARTS[str] = line_starts
return line_starts
end
Files.get_line_number = function(str, pos)
local line_starts = Files.get_line_starts(str)
local lo, hi = 1, #line_starts
while lo <= hi do
local mid = math.floor((lo + hi) / 2)
if line_starts[mid] > pos then
hi = mid - 1
else
lo = mid + 1
end
end
return hi
end
Files.get_line = function(str, line_no)
local line_starts = Files.get_line_starts(str)
local start = line_starts[line_no]
if not (start) then
return
end
local stop = line_starts[line_no + 1]
if not (stop) then
return
end
return str:sub(start, stop - 2)
end
2018-07-17 16:13:35 -07:00
local get_lines = re.compile([[ lines <- {| line (%nl line)* |}
line <- {[^%nl]*}
]], {
nl = lpeg.P("\r") ^ -1 * lpeg.P("\n")
})
Files.get_lines = function(str)
return get_lines:match(str)
end
return Files