2018-06-23 17:22:23 -07:00
-- Some file utilities for searching for files recursively and using package.nomsupath
2018-06-24 23:18:32 -07:00
lpeg = require 'lpeg'
re = require 're'
2018-07-21 14:43:49 -07:00
Files = {}
2018-07-24 15:08:44 -07:00
assert package.nomsupath, "No package.nomsupath was found"
2018-06-23 17:22:23 -07:00
2018-07-24 16:43:06 -07:00
run_cmd = (cmd)->
f = io.popen(cmd)
lines = [line for line in f\lines!]
return nil unless f\close!
return lines
2018-07-21 14:43:49 -07:00
_SPOOFED_FILES = {}
_FILE_CACHE = setmetatable {}, __index:_SPOOFED_FILES
_BROWSE_CACHE = {}
2018-06-23 17:22:23 -07:00
-- Create a fake file and put it in the cache
2018-11-08 15:23:22 -08:00
_anon_number = 0
2018-07-21 14:43:49 -07:00
Files.spoof = (filename, contents)->
2018-11-08 15:23:22 -08:00
if not contents
filename, contents = "<anonymous file ##{_anon_number}>", filename
_anon_number += 1
2018-07-21 14:43:49 -07:00
_SPOOFED_FILES[filename] = contents
2018-11-08 15:23:22 -08:00
return filename
2018-06-23 17:22:23 -07:00
-- Read a file's contents (searching first locally, then in the nomsupath)
2018-07-21 14:43:49 -07:00
Files.read = (filename)->
2018-06-23 17:22:23 -07:00
if file_contents = _FILE_CACHE[filename]
return file_contents
2018-06-28 14:12:24 -07:00
if filename == 'stdin'
2018-11-08 15:23:22 -08:00
contents = io.read('*a')
Files.spoof('stdin', contents)
return contents
2018-06-23 17:22:23 -07:00
file = io.open(filename)
return nil unless file
contents = file\read("*a")
file\close!
_FILE_CACHE[filename] = contents
return contents
2018-06-23 17:24:28 -07:00
{:match, :gsub} = string
2018-07-17 16:13:35 -07:00
-- TODO: improve sanitization
sanitize = (path)->
path = gsub(path,"\\","\\\\")
path = gsub(path,"`","")
path = gsub(path,'"','\\"')
path = gsub(path,"$","")
return path
2018-07-21 14:43:49 -07:00
Files.exists = (path)->
return true if _SPOOFED_FILES[path]
2018-07-24 17:36:45 -07:00
return true if path == 'stdin'
2018-07-24 16:43:06 -07:00
return true if run_cmd("ls #{sanitize(path)}")
2018-07-24 15:08:44 -07:00
for nomsupath in package.nomsupath\gmatch("[^;]+")
2018-07-24 16:43:06 -07:00
return true if run_cmd("ls #{nomsupath}/#{sanitize(path)}")
2018-07-17 16:13:35 -07:00
return false
browse = (path)->
2018-07-21 14:43:49 -07:00
unless _BROWSE_CACHE[path]
local files
_BROWSE_CACHE[path] = if _SPOOFED_FILES[path]
{path}
2018-07-24 16:43:06 -07:00
else run_cmd('find -L "'..path..'" -not -path "*/\\.*" -type f') or false
2018-07-21 14:43:49 -07:00
return _BROWSE_CACHE[path]
2018-07-17 16:13:35 -07:00
2018-06-23 17:22:23 -07:00
ok, lfs = pcall(require, "lfs")
if ok
2018-07-09 16:58:46 -07:00
raw_file_exists = (filename)->
mode = lfs.attributes(filename, 'mode')
2018-07-09 19:22:40 -07:00
return if mode == 'file' or mode == 'directory' or mode == 'link' then true else false
2018-07-21 14:43:49 -07:00
Files.exists = (path)->
return true if _SPOOFED_FILES[path]
2018-07-09 16:58:46 -07:00
return true if path == 'stdin' or raw_file_exists(path)
2018-07-24 15:08:44 -07:00
for nomsupath in package.nomsupath\gmatch("[^;]+")
return true if raw_file_exists(nomsupath.."/"..path)
2018-07-09 16:58:46 -07:00
return false
2018-07-09 19:22:40 -07:00
2018-07-17 16:13:35 -07:00
export browse
browse = (filename)->
2018-07-21 14:43:49 -07:00
unless _BROWSE_CACHE[filename]
2018-07-24 17:36:45 -07:00
_BROWSE_CACHE[filename] = if _SPOOFED_FILES[filename] or filename == 'stdin'
2018-07-21 14:43:49 -07:00
{filename}
else
file_type, err = lfs.attributes(filename, 'mode')
2018-07-24 16:43:06 -07:00
switch file_type
when "file", "char device"
2018-07-21 14:43:49 -07:00
{filename}
2018-07-24 16:43:06 -07:00
when "directory", "link"
files = {}
for subfile in lfs.dir(filename)
continue if subfile == "." or subfile == ".."
for f in *(browse(filename.."/"..subfile) or {})
files[#files+1] = f
files
2018-07-21 14:43:49 -07:00
else false
return _BROWSE_CACHE[filename]
2018-06-23 17:22:23 -07:00
else
2018-07-24 16:43:06 -07:00
unless run_cmd('find . -maxdepth 0')
2018-07-24 17:17:03 -07:00
url = if jit
'https://github.com/spacewander/luafilesystem'
else 'https://github.com/keplerproject/luafilesystem'
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: #{url} or `luarocks install luafilesystem`)", 0
2018-07-09 16:58:46 -07:00
2018-07-21 14:43:49 -07:00
Files.walk = (path, flush_cache=false)->
2018-07-17 16:13:35 -07:00
if flush_cache
2018-07-21 14:43:49 -07:00
export _BROWSE_CACHE
_BROWSE_CACHE = {}
2018-07-24 15:08:44 -07:00
local files
2018-07-26 14:29:23 -07:00
if path == 'stdin' or _SPOOFED_FILES[path]
2018-07-24 17:36:45 -07:00
files = {path}
2018-11-02 14:38:24 -07:00
elseif path\match("^[~/]") or path\match("^%./") or path\match("^%.%./")
files = browse(path)
2018-07-24 17:36:45 -07:00
else
for nomsupath in package.nomsupath\gmatch("[^;]+")
if files = browse(nomsupath.."/"..path) then break
2018-10-31 03:51:37 -07:00
files or= {}
files = [gsub(f, "^%./", "") for f in *files]
return ipairs(files)
2018-06-23 17:22:23 -07:00
2018-06-24 23:18:32 -07:00
line_counter = re.compile([[
lines <- {| line (%nl line)* |}
line <- {} (!%nl .)*
]], nl:lpeg.P("\r")^-1 * lpeg.P("\n"))
-- LINE_STARTS is a mapping from strings to a table that maps line number to character positions
_LINE_STARTS = {}
2018-07-21 14:43:49 -07:00
Files.get_line_starts = (str)->
2018-06-24 23:18:32 -07:00
if type(str) != 'string'
str = tostring(str)
if starts = _LINE_STARTS[str]
return starts
line_starts = line_counter\match(str)
_LINE_STARTS[str] = line_starts
return line_starts
2018-07-21 14:43:49 -07:00
Files.get_line_number = (str, pos)->
line_starts = Files.get_line_starts(str)
2018-06-24 23:18:32 -07:00
-- Binary search for line number of position
lo, hi = 1, #line_starts
while lo <= hi
mid = math.floor((lo+hi)/2)
if line_starts[mid] > pos
hi = mid-1
else lo = mid+1
return hi
2018-07-21 14:43:49 -07:00
Files.get_line = (str, line_no)->
line_starts = Files.get_line_starts(str)
2018-07-22 14:56:12 -07:00
start = line_starts[line_no]
return unless start
stop = line_starts[line_no+1]
return unless stop
2018-11-09 14:36:15 -08:00
return (str\sub(start, stop - 2))
2018-06-26 15:52:38 -07:00
2018-07-17 16:13:35 -07:00
get_lines = re.compile([[
lines <- {| line (%nl line)* |}
line <- {[^%nl]*}
]], nl:lpeg.P("\r")^-1 * lpeg.P("\n"))
2018-07-21 14:43:49 -07:00
Files.get_lines = (str)-> get_lines\match(str)
2018-06-26 15:52:38 -07:00
2018-07-21 14:43:49 -07:00
return Files