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 = { }
2018-07-24 16:43:06 -07:00
local run_cmd
run_cmd = function ( cmd )
2019-01-10 16:33:37 -08:00
local f = io.popen ( cmd .. ' 2>/dev/null ' )
2018-07-24 16:43:06 -07:00
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
2018-07-21 14:43:49 -07:00
local _SPOOFED_FILES = { }
local _BROWSE_CACHE = { }
2018-11-08 15:23:22 -08:00
local _anon_number = 0
2018-07-21 14:43:49 -07:00
Files.spoof = function ( filename , contents )
2018-11-08 15:23:22 -08:00
if not contents then
filename , contents = " <anonymous file # " .. tostring ( _anon_number ) .. " > " , filename
_anon_number = _anon_number + 1
end
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
end
2019-01-08 16:38:34 -08:00
Files.read = function ( filename )
do
local contents = _SPOOFED_FILES [ filename ]
if contents then
return contents
2018-06-23 17:22:23 -07:00
end
end
2018-11-20 14:52:59 -08:00
if filename == ' stdin ' or filename == ' - ' then
2018-11-08 15:23:22 -08:00
local contents = io.read ( ' *a ' )
Files.spoof ( ' stdin ' , contents )
2018-11-20 14:52:59 -08:00
Files.spoof ( ' - ' , contents )
2018-11-08 15:23:22 -08:00
return contents
2018-06-28 14:12:24 -07:00
end
2018-06-23 17:22:23 -07:00
local file = io.open ( filename )
if not ( file ) then
return nil
end
local contents = file : read ( " *a " )
file : close ( )
2018-11-20 14:52:59 -08:00
return contents or nil
2018-06-23 17:22:23 -07:00
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-11-20 14:52:59 -08:00
if path == ' stdin ' or path == ' - ' then
2018-07-24 17:36:45 -07:00
return true
end
2018-07-24 16:43:06 -07:00
if run_cmd ( " ls " .. tostring ( sanitize ( path ) ) ) then
2018-07-17 16:13:35 -07:00
return true
end
return false
end
2018-11-19 17:21:08 -08:00
Files.list = function ( path )
2018-07-21 14:43:49 -07:00
if not ( _BROWSE_CACHE [ path ] ) then
local files
2018-11-20 14:52:59 -08:00
if _SPOOFED_FILES [ path ] or path == ' stdin ' or path == ' - ' then
2018-07-21 14:43:49 -07:00
_BROWSE_CACHE [ path ] = {
path
}
else
2019-01-10 16:33:37 -08:00
_BROWSE_CACHE [ path ] = run_cmd ( ' find -L " ' .. path .. ' " -not -path "*/ \\ .*" -type f ' ) 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
2019-01-27 02:01:57 -08:00
Files.make_directory = function ( path )
return run_cmd ( ' mkdir ' .. path )
end
2018-06-23 17:22:23 -07:00
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-11-20 14:52:59 -08:00
if path == ' stdin ' or path == ' - ' or raw_file_exists ( path ) then
2018-07-09 16:58:46 -07:00
return true
end
return false
end
2018-11-19 17:21:08 -08:00
Files.list = function ( path )
if not ( _BROWSE_CACHE [ path ] ) then
2018-11-20 14:52:59 -08:00
if _SPOOFED_FILES [ path ] or path == ' stdin ' or path == ' - ' then
2018-11-19 17:21:08 -08:00
_BROWSE_CACHE [ path ] = {
path
2018-07-17 16:13:35 -07:00
}
2018-07-21 14:43:49 -07:00
else
2018-11-19 17:21:08 -08:00
local file_type , err = lfs.attributes ( path , ' mode ' )
2018-07-24 16:43:06 -07:00
local _exp_0 = file_type
if " file " == _exp_0 or " char device " == _exp_0 then
2018-11-19 17:21:08 -08:00
_BROWSE_CACHE [ path ] = {
path
2018-07-21 14:43:49 -07:00
}
2018-07-24 16:43:06 -07:00
elseif " directory " == _exp_0 or " link " == _exp_0 then
2018-07-21 14:43:49 -07:00
local files = { }
2018-11-19 17:21:08 -08:00
for subfile in lfs.dir ( path ) do
2018-07-24 15:08:44 -07:00
local _continue_0 = false
repeat
if subfile == " . " or subfile == " .. " then
_continue_0 = true
break
end
2018-11-19 17:21:08 -08:00
local _list_0 = ( Files.list ( path .. " / " .. subfile ) or { } )
2018-07-21 14:43:49 -07:00
for _index_0 = 1 , # _list_0 do
local f = _list_0 [ _index_0 ]
files [ # files + 1 ] = f
end
2018-07-24 15:08:44 -07:00
_continue_0 = true
until true
if not _continue_0 then
break
2018-07-17 16:13:35 -07:00
end
2018-06-23 17:22:23 -07:00
end
2018-11-19 17:21:08 -08:00
_BROWSE_CACHE [ path ] = files
2018-07-21 14:43:49 -07:00
else
2018-11-19 17:21:08 -08:00
_BROWSE_CACHE [ path ] = false
end
end
if _BROWSE_CACHE [ path ] then
for i , f in ipairs ( _BROWSE_CACHE [ path ] ) do
if f : match ( " ^%./ " ) then
_BROWSE_CACHE [ path ] [ i ] = f : sub ( 3 )
end
2018-06-23 17:22:23 -07:00
end
end
2018-07-17 16:13:35 -07:00
end
2018-11-19 17:21:08 -08:00
return _BROWSE_CACHE [ path ]
2018-06-23 17:22:23 -07:00
end
2019-01-27 02:01:57 -08:00
Files.make_directory = lfs.mkdir
2018-06-23 17:22:23 -07:00
else
2019-01-10 16:33:37 -08:00
if not ( run_cmd ( ' find . -maxdepth 0 ' ) ) then
2018-07-24 17:17:03 -07:00
local url
if jit then
url = ' https://github.com/spacewander/luafilesystem '
else
url = ' https://github.com/keplerproject/luafilesystem '
end
2019-01-27 02:01:57 -08: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: " .. tostring ( url ) .. " or `luarocks install luafilesystem`) \n " .. tostring ( lfs ) .. " \n package.cpath: " .. tostring ( package.cpath ) , 0 )
2018-06-23 17:22:23 -07:00
end
2018-07-17 16:13:35 -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-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
2018-11-09 14:36:15 -08:00
return ( str : sub ( start , stop - 2 ) )
2018-06-26 15:52:38 -07:00
end
2018-07-21 14:43:49 -07:00
return Files