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-06-23 17:22:23 -07:00
files = {}
_FILE_CACHE = {}
-- Create a fake file and put it in the cache
files.spoof = (filename, contents)->
_FILE_CACHE[filename] = contents
-- Read a file's contents (searching first locally, then in the nomsupath)
files.read = (filename)->
if file_contents = _FILE_CACHE[filename]
return file_contents
2018-06-28 14:12:24 -07:00
if filename == 'stdin'
contents = io.read('*a')
_FILE_CACHE['stdin'] = contents
return contents
2018-06-23 17:22:23 -07:00
file = io.open(filename)
if package.nomsupath and not file
for nomsupath in package.nomsupath\gmatch("[^;]+")
file = io.open(nomsupath.."/"..filename)
break if file
return nil unless file
contents = file\read("*a")
file\close!
_FILE_CACHE[filename] = contents
return contents
iterate_single = (item, prev) -> if item == prev then nil else item
-- `walk` returns an iterator over all files matching a path.
2018-06-23 17:24:28 -07:00
{:match, :gsub} = string
2018-06-23 17:22:23 -07:00
iterate_single = (item, prev) -> if item == prev then nil else item
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-09 16:58:46 -07:00
files.exists = (path)->
return true if path == 'stdin' or raw_file_exists(path)
if package.nomsupath
for nomsupath in package.nomsupath\gmatch("[^;]+")
return true if raw_file_exists(nomsupath.."/"..path)
return false
-- Return 'true' if any files or directories are found, otherwise 'false', and yield every filename
browse = (filename)->
2018-07-09 19:22:40 -07:00
file_type, err = lfs.attributes(filename, 'mode')
2018-07-09 16:58:46 -07:00
if file_type == 'file'
if match(filename, "%.nom$") or match(filename, "%.lua$")
coroutine.yield filename
return true
2018-07-09 19:22:40 -07:00
elseif file_type == 'directory' or file_type == 'link'
2018-07-09 16:58:46 -07:00
for subfile in lfs.dir(filename)
2018-07-09 19:22:40 -07:00
unless subfile == "." or subfile == ".."
2018-07-09 16:58:46 -07:00
browse(filename.."/"..subfile)
return true
elseif file_type == 'char device'
coroutine.yield(filename)
return true
2018-07-09 19:22:40 -07:00
2018-07-09 16:58:46 -07:00
return false
2018-06-23 17:22:23 -07:00
files.walk = (path)->
2018-06-28 14:12:24 -07:00
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin'
return iterate_single, path
2018-06-23 17:22:23 -07:00
return coroutine.wrap ->
if not browse(path) and package.nomsupath
for nomsupath in package.nomsupath\gmatch("[^;]+")
break if browse(nomsupath.."/"..path)
return nil
else
2018-06-24 18:39:17 -07:00
if io.popen('find . -maxdepth 0')\close!
2018-06-23 17:22:23 -07:00
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: http://keplerproject.github.io/luafilesystem/ or `luarocks install luafilesystem`)", 0
2018-07-09 16:58:46 -07:00
-- TODO: improve sanitization
sanitize = (path)->
2018-06-23 17:22:23 -07:00
path = gsub(path,"\\","\\\\")
path = gsub(path,"`","")
path = gsub(path,'"','\\"')
path = gsub(path,"$","")
2018-07-09 16:58:46 -07:00
return path
files.exists = (path)->
return true unless io.popen("ls #{sanitize(path)}")\close!
if package.nomsupath
for nomsupath in package.nomsupath\gmatch("[^;]+")
return true unless io.popen("ls #{nomsupath}/#{sanitize(path)}")\close!
return false
files.walk = (path)->
if match(path, "%.nom$") or match(path, "%.lua$") or path == 'stdin'
return iterate_single, path
path = sanitize(path)
2018-06-23 17:22:23 -07:00
return coroutine.wrap ->
f = io.popen('find -L "'..path..'" -not -path "*/\\.*" -type f -name "*.nom"')
found = false
for line in f\lines!
found = true
coroutine.yield(line)
if not found and package.nomsupath
f\close!
for nomsupath in package.nomsupath\gmatch("[^;]+")
f = io.popen('find -L "'..package.nomsupath..'/'..path..'" -not -path "*/\\.*" -type f -name "*.nom"')
for line in f\lines!
found = true
coroutine.yield(line)
f\close!
break if found
unless found
error("Invalid file path: "..tostring(path))
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"))
get_lines = 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 = {}
files.get_line_starts = (str)->
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
files.get_line_number = (str, pos)->
line_starts = files.get_line_starts(str)
-- 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-06-26 15:52:38 -07:00
files.get_line = (str, line_no)->
line_starts = files.get_line_starts(str)
2018-06-28 14:12:24 -07:00
return str\sub(line_starts[line_no] or 1, (line_starts[line_no+1] or 1) - 2)
2018-06-26 15:52:38 -07:00
files.get_lines = (str)-> get_lines\match(str)
2018-06-23 17:22:23 -07:00
return files