2018-06-24 23:18:32 -07:00
local lpeg = require ( ' lpeg ' )
local re = require ( ' re ' )
2018-07-21 14:43:49 -07:00
local Files = { }
local _SPOOFED_FILES = { }
local _FILE_CACHE = setmetatable ( { } , {
__index = _SPOOFED_FILES
} )
local _BROWSE_CACHE = { }
Files.spoof = function ( filename , contents )
_SPOOFED_FILES [ filename ] = contents
return contents
2018-06-23 17:22:23 -07:00
end
2018-07-21 14:43:49 -07:00
Files.read = function ( filename )
2018-06-23 17:22:23 -07:00
do
local file_contents = _FILE_CACHE [ filename ]
if file_contents then
return file_contents
end
end
2018-06-28 14:12:24 -07:00
if filename == ' stdin ' then
2018-07-21 14:43:49 -07:00
return Files.spoof ( ' stdin ' , io.read ( ' *a ' ) )
2018-06-28 14:12:24 -07:00
end
2018-06-23 17:22:23 -07:00
local file = io.open ( filename )
if package.nomsupath and not file then
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
file = io.open ( nomsupath .. " / " .. filename )
if file then
break
end
end
end
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
2018-06-23 17:22:23 -07:00
do
local _obj_0 = string
2018-06-23 17:24:28 -07:00
match , gsub = _obj_0.match , _obj_0.gsub
2018-06-23 17:22:23 -07:00
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
2018-07-21 14:43:49 -07:00
Files.exists = function ( path )
if _SPOOFED_FILES [ path ] then
return true
end
2018-07-17 16:13:35 -07:00
if not ( io.popen ( " ls " .. tostring ( sanitize ( path ) ) ) : close ( ) ) then
return true
end
if package.nomsupath then
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
if not ( io.popen ( " ls " .. tostring ( nomsupath ) .. " / " .. tostring ( sanitize ( path ) ) ) : close ( ) ) then
return true
end
end
2018-06-23 17:22:23 -07:00
end
2018-07-17 16:13:35 -07:00
return false
end
local browse
browse = function ( path )
2018-07-21 14:43:49 -07:00
if not ( _BROWSE_CACHE [ path ] ) then
local files
if _SPOOFED_FILES [ path ] then
_BROWSE_CACHE [ path ] = {
path
}
else
local f = io.popen ( ' find -L " ' .. package.nomsupath .. ' / ' .. path .. ' " -not -path "*/ \\ .*" -type f -name "*.nom" ' )
do
local _tbl_0 = { }
for line in f : lines ( ) do
local _key_0 , _val_0 = line
_tbl_0 [ _key_0 ] = _val_0
end
files = _tbl_0
2018-07-17 16:13:35 -07:00
end
2018-07-21 14:43:49 -07:00
_BROWSE_CACHE [ path ] = f : close ( ) and files or false
2018-07-17 16:13:35 -07:00
end
end
2018-07-21 14:43:49 -07:00
return _BROWSE_CACHE [ path ]
2018-06-23 17:22:23 -07:00
end
local ok , lfs = pcall ( require , " lfs " )
2018-06-24 23:18:32 -07:00
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
2018-06-28 14:12:24 -07:00
end
2018-07-09 16:58:46 -07:00
end
2018-07-21 14:43:49 -07:00
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
if package.nomsupath then
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
if raw_file_exists ( nomsupath .. " / " .. path ) then
2018-06-23 17:22:23 -07:00
return true
end
2018-07-09 16:58:46 -07:00
end
end
return false
end
browse = function ( filename )
2018-07-21 14:43:49 -07:00
if not ( _BROWSE_CACHE [ filename ] ) then
if _SPOOFED_FILES [ filename ] then
_BROWSE_CACHE [ filename ] = {
2018-07-17 16:13:35 -07:00
filename
}
2018-07-21 14:43:49 -07:00
else
local file_type , err = lfs.attributes ( filename , ' mode ' )
if file_type == ' file ' then
if match ( filename , " %.nom$ " ) or match ( filename , " %.lua$ " ) then
_BROWSE_CACHE [ filename ] = {
filename
}
else
_BROWSE_CACHE [ filename ] = false
end
elseif file_type == ' char device ' then
_BROWSE_CACHE [ filename ] = {
filename
}
elseif file_type == ' directory ' or file_type == ' link ' then
local files = { }
for subfile in lfs.dir ( filename ) do
if not ( subfile == " . " or subfile == " .. " ) then
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
2018-07-17 16:13:35 -07:00
end
2018-06-23 17:22:23 -07:00
end
2018-07-21 14:43:49 -07:00
_BROWSE_CACHE [ filename ] = files
else
_BROWSE_CACHE [ filename ] = false
2018-06-23 17:22:23 -07:00
end
end
2018-07-17 16:13:35 -07:00
end
2018-07-21 14:43:49 -07:00
return _BROWSE_CACHE [ filename ]
2018-06-23 17:22:23 -07:00
end
else
2018-06-24 18:39:17 -07:00
if io.popen ( ' find . -maxdepth 0 ' ) : close ( ) then
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 )
end
2018-07-17 16:13:35 -07:00
end
2018-07-21 14:43:49 -07:00
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
2018-07-21 14:43:49 -07:00
_BROWSE_CACHE = { }
2018-07-09 16:58:46 -07:00
end
2018-07-21 14:43:49 -07:00
local files = browse ( path )
if package.nomsupath and not files then
2018-07-17 16:13:35 -07:00
for nomsupath in package.nomsupath : gmatch ( " [^;]+ " ) do
do
2018-07-21 14:43:49 -07:00
files = browse ( nomsupath .. " / " .. path )
if files then
2018-07-17 16:13:35 -07:00
break
2018-07-09 16:58:46 -07:00
end
end
end
end
2018-07-17 16:13:35 -07:00
local iter
2018-07-21 14:43:49 -07:00
iter = function ( files , i )
if not ( files ) then
2018-07-19 20:41:31 -07:00
return
end
2018-07-17 16:13:35 -07:00
i = i + 1
do
2018-07-21 14:43:49 -07:00
local f = files [ i ]
2018-07-17 16:13:35 -07:00
if f then
return i , f
2018-06-23 17:22:23 -07:00
end
2018-07-17 16:13:35 -07:00
end
2018-06-23 17:22:23 -07:00
end
2018-07-21 14:43:49 -07:00
return iter , files , 0
2018-06-23 17:22:23 -07:00
end
2018-06-24 23:18:32 -07:00
local line_counter = re.compile ( [ [ lines <- { | line ( % nl line ) * | }
line <- { } ( ! % nl . ) *
] ] , {
nl = lpeg.P ( " \r " ) ^ - 1 * lpeg.P ( " \n " )
} )
local _LINE_STARTS = { }
2018-07-21 14:43:49 -07:00
Files.get_line_starts = function ( str )
2018-06-24 23:18:32 -07:00
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
2018-07-17 16:13:35 -07:00
local log = { }
2018-07-21 14:43:49 -07:00
Files.get_line_number = function ( str , pos )
local line_starts = Files.get_line_starts ( str )
2018-06-24 23:18:32 -07:00
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
2018-07-21 14:43:49 -07:00
Files.get_line = function ( str , line_no )
local line_starts = Files.get_line_starts ( str )
2018-07-22 14:57:37 -07:00
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 )
2018-06-26 15:52:38 -07:00
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 " )
} )
2018-07-21 14:43:49 -07:00
Files.get_lines = function ( str )
2018-06-26 15:52:38 -07:00
return get_lines : match ( str )
end
2018-07-21 14:43:49 -07:00
return Files