-------------------------------------------------------------------------------
-- Useful functions for getting directory contents and matching them against wildcards
local lfs = require 'lfs'
local utils = require 'pl.utils'
local path = require 'pl.path'
local is_windows = path.is_windows
local tablex = require 'pl.tablex'
local attrib = lfs.attributes
local ldir = lfs.dir
local chdir = lfs.chdir
local mkdir = lfs.mkdir
local escape = utils.escape
local sub = string.sub
local os,pcall,ipairs,pairs,require,setmetatable,_G = os,pcall,ipairs,pairs,require,setmetatable,_G
local remove = os.remove
local append = table.insert
local print = print
local assert = assert
local type = type
local wrap = coroutine.wrap
local yield = coroutine.yield
local assert_arg,assert_string,raise = utils.assert_arg,utils.assert_string,utils.raise
local List = utils.stdmt.List
module ('pl.dir',utils._module)
local function assert_dir (n,val)
assert_arg(n,val,'string',path.isdir,'not a directory')
end
local function assert_file (n,val)
assert_arg(n,val,'string',path.isfile,'not a file')
end
local function filemask(mask)
mask = escape(mask)
return mask:gsub('%%%*','.+'):gsub('%%%?','.')..'$'
end
--- does the filename match the shell pattern?.
-- (cf. fnmatch.fnmatch in Python, 11.8)
-- @param file A file name
-- @param pattern A shell pattern
function fnmatch(file,pattern)
assert_string(1,file)
assert_string(2,pattern)
return path.normcase(file):find(filemask(pattern)) ~= nil
end
--- return a list of all files in a list of files which match the pattern.
-- (cf. fnmatch.filter in Python, 11.8)
-- @param files A table containing file names
-- @param pattern A shell pattern.
function filter(files,pattern)
assert_arg(1,files,'table')
assert_string(2,pattern)
local res = {}
local mask = filemask(pattern)
for i,f in ipairs(files) do
if f:find(mask) then append(res,f) end
end
return setmetatable(res,List)
end
function _listfiles(dir,filemode,match)
local res = {}
if not dir then dir = '.' end
for f in ldir(dir) do
if f ~= '.' and f ~= '..' then
local p = path.join(dir,f)
local mode = attrib(p,'mode')
if mode == filemode and (not match or match(p)) then
append(res,p)
end
end
end
return setmetatable(res,List)
end
--- return a list of all files in a directory which match the a shell pattern.
-- @param dir A directory. If not given, all files in current directory are returned.
-- @param mask A shell pattern. If not given, all files are returned.
function getfiles(dir,mask)
assert_dir(1,dir)
assert_string(2,mask)
local match
if mask then
mask = filemask(mask)
match = function(f)
return f:find(mask)
end
end
return _listfiles(dir,'file',match)
end
--- return a list of all subdirectories of the directory.
-- @param dir A directory
function getdirectories(dir)
assert_dir(1,dir)
return _listfiles(dir,'directory')
end
local function quote_if_necessary (f)
if f:find '%s' then
return '"'..f..'"'
else
return f
end
end
local alien,no_alien,kernel,CopyFile,MoveFile,GetLastError,win32_errors,cmd_tmpfile
local function file_op (is_copy,src,dest,flag)
local null
if is_windows then
local res
-- if we haven't tried to load Alien before, then do so
if not alien and not no_alien then
res,alien = pcall(require,'alien')
no_alien = not res
if no_alien then alien = nil end
if alien then
-- register the Win32 CopyFile and MoveFile functions
local copySpec = {'string','string','int',ret='int',abi='stdcall'}
kernel = alien.load('kernel32.dll')
CopyFile = kernel.CopyFileA
CopyFile:types(copySpec)
local moveSpec = {'string','string',ret='int',abi='stdcall'}
MoveFile = kernel.MoveFileA
MoveFile:types(moveSpec)
GetLastError = kernel.GetLastError
GetLastError:types{ret ='int', abi='stdcall'}
win32_errors = {
ERROR_FILE_NOT_FOUND = 2,
ERROR_PATH_NOT_FOUND = 3,
ERROR_ACCESS_DENIED = 5,
ERROR_WRITE_PROTECT = 19,
ERROR_BAD_UNIT = 20,
ERROR_NOT_READY = 21,
ERROR_WRITE_FAULT = 29,
ERROR_READ_FAULT = 30,
ERROR_SHARING_VIOLATION = 32,
ERROR_LOCK_VIOLATION = 33,
ERROR_HANDLE_DISK_FULL = 39,
ERROR_BAD_NETPATH = 53,
ERROR_NETWORK_BUSY = 54,
ERROR_DEV_NOT_EXIST = 55,
ERROR_FILE_EXISTS = 80,
ERROR_OPEN_FAILED = 110,
ERROR_INVALID_NAME = 123,
ERROR_BAD_PATHNAME = 161,
ERROR_ALREADY_EXISTS = 183,
}
end
end
if not cmd_tmpfile then cmd_tmpfile = path.tmpname () end
-- fallback if there's no Alien, just use DOS commands *shudder*
if not CopyFile then
src = path.normcase(src)
dest = path.normcase(dest)
cmd = is_copy and 'copy' or 'rename'
null = ' > '..cmd_tmpfile
else
if is_copy then ret = CopyFile(src,dest,flag)
else ret = MoveFile(src,dest) end
if ret == 0 then
local err = GetLastError()
for name,value in pairs(win32_errors) do
if value == err then return false,name end
end
return false,"Error #"..err
else return true
end
end
else -- for Unix, just use cp for now
if not cmd_tmpfile then cmd_tmpfile = path.tmpname () end
cmd = is_copy and 'cp' or 'mv'
null = ' 2> '..cmd_tmpfile
end
if flag == 1 and path.exists(dest) then
return false,"cannot overwrite destination"
end
src = quote_if_necessary(src)
dest = quote_if_necessary(dest)
-- let's make this as quiet a call as we can...
cmd = cmd..' '..src..' '..dest..null
--print(cmd)
local ret = os.execute(cmd) == 0
if not ret then
return false,(utils.readfile(cmd_tmpfile):gsub('\n(.*)',''))
else
return true
end
end
--- copy a file.
-- @param src source file
-- @param dest destination file
-- @param flag true if you want to force the copy (default)
-- @return true if operation succeeded
function copyfile (src,dest,flag)
assert_string(1,src)
assert_string(2,dest)
flag = flag==nil or flag
return file_op(true,src,dest,flag and 0 or 1)
end
--- move a file.
-- @param src source file
-- @param dest destination file
-- @return true if operation succeeded
function movefile (src,dest)
assert_string(1,src)
assert_string(2,dest)
return file_op(false,src,dest,0)
end
local function _dirfiles(dir)
local dirs = {}
local files = {}
for f in ldir(dir) do
if f ~= '.' and f ~= '..' then
local p = path.join(dir,f)
local mode = attrib(p,'mode')
if mode=='directory' then
append(dirs,f)
else
append(files,f)
end
end
end
return setmetatable(dirs,List),setmetatable(files,List)
end
local function _walker(root,bottom_up)
local dirs,files = _dirfiles(root)
if not bottom_up then yield(root,dirs,files) end
for i,d in ipairs(dirs) do
_walker(root..path.sep..d,bottom_up)
end
if bottom_up then yield(root,dirs,files) end
end
--- return an iterator which walks through a directory tree starting at root.
-- The iterator returns (root,dirs,files)
-- Note that dirs and files are lists of names (i.e. you must say _path.join(root,d)_
-- to get the actual full path)
-- If bottom_up is false (or not present), then the entries at the current level are returned
-- before we go deeper. This means that you can modify the returned list of directories before
-- continuing.
-- This is a clone of os.walk from the Python libraries.
-- @param root A starting directory
-- @param bottom_up False if we start listing entries immediately.
function walk(root,bottom_up)
assert_string(1,root)
if not path.isdir(root) then return raise 'not a directory' end
return wrap(function () _walker(root,bottom_up) end)
end
--- remove a whole directory tree.
-- @param fullpath A directory path
function rmtree(fullpath)
assert_string(1,fullpath)
if not path.isdir(fullpath) then return raise 'not a directory' end
for root,dirs,files in walk(fullpath,true) do
for i,f in ipairs(files) do
remove(path.join(root,f))
end
lfs.rmdir(root)
end
end
local dirpat
if path.is_windows then
dirpat = '(.+)\\[^\\]+$'
else
dirpat = '(.+)/[^/]+$'
end
function _makepath(p)
-- windows root drive case
if p:find '^%a:$' then
return true
end
if not path.isdir(p) then
local subp = p:match(dirpat)
if not _makepath(subp) then return raise ('cannot create '..subp) end
--print('create',p)
return lfs.mkdir(p)
else
return true
end
end
--- create a directory path.
-- This will create subdirectories as necessary!
-- @param path A directory path
function makepath (path)
assert_string(1,path)
return _makepath(path.normcase(path.abspath(path)))
end
--- clone a directory tree. Will always try to create a new directory structure
-- if necessary.
-- @param path1 the base path of the source tree
-- @param path2 the new base path for the destination
-- @param file_fun an optional function to apply on all files
-- @param verbose an optional boolean to control the verbosity of the output.
-- @return if failed, false plus an error message. If completed the traverse,
-- true, a list of failed directory creations and a list of failed file operations.
-- @usage clonetree('.','../backup',copyfile)
function clonetree (path1,path2,file_fun,verbose)
assert_string(1,path1)
assert_string(2,path2)
local abspath,normcase,isdir,join = path.abspath,path.normcase,path.isdir,path.join
local faildirs,failfiles = {},{}
if not isdir(path1) then return raise 'source is not a valid directory' end
path1 = abspath(normcase(path1))
path2 = abspath(normcase(path2))
if verbose then verbose('normalized:',path1,path2) end
-- particularly NB that the new path isn't fully contained in the old path
if path1 == path2 then return raise "paths are the same" end
local i1,i2 = path2:find(path1,1,true)
if i2 == #path1 and path2:sub(i2+1,i2+1) == path.sep then
return raise 'destination is a subdirectory of the source'
end
local cp = path.common_prefix (path1,path2)
local idx = #cp
if idx == 0 then -- no common path, but watch out for Windows paths!
if path1:sub(2,2) == ':' then idx = 3 end
end
for root,dirs,files in walk(path1) do
local opath = path2..root:sub(idx)
if verbose then verbose('paths:',opath,root) end
if not isdir(opath) then
local ret = makepath(opath)
if not ret then append(faildirs,opath) end
if verbose then verbose('creating:',opath,ret) end
end
if file_fun then
for i,f in ipairs(files) do
local p1 = join(root,f)
local p2 = join(opath,f)
local ret = file_fun(p1,p2)
if not ret then append(failfiles,p2) end
if verbose then
verbose('files:',p1,p2,ret)
end
end
end
end
return true,faildirs,failfiles
end
--- Recursively returns all the file starting at path. It can optionally take a shell pattern and
-- only returns files that match pattern. If a pattern is given it will do a case insensitive search.
-- @param path {string} A directory. If not given, all files in current directory are returned.
-- @param pattern {string} A shell pattern. If not given, all files are returned.
-- @return Table containing all the files found recursively starting at path and filtered by pattern.
function getallfiles( path, pattern )
assert( type( path ) == "string", "bad argument #1 to 'GetAllFiles' (Expected string but recieved " .. type( path ) .. ")" )
pattern = pattern or ""
function dirtree( dir )
assert( dir and dir ~= "", "directory parameter is missing or empty" )
if sub( dir, -1 ) == "/" then
dir = sub( dir, 1, -2 )
end
local function yieldtree( dir )
for entry in ldir( dir ) do
if entry ~= "." and entry ~= ".." then
entry = dir .. "/" .. entry
local attr = attrib( entry )
if attr then -- Just in case a symlink is broken.
yield( entry, attr )
if attr.mode == "directory" then
yieldtree( entry )
end
end
end
end
end
return wrap( function() yieldtree( dir ) end )
end
local files = {}
for filename, attr in dirtree( path ) do
if "file" == attr.mode then
local mask = filemask( pattern ):lower()
if filename:lower():find( mask ) then
files[#files + 1] = filename
end
end
end
return files
end