2017-02-13 15:16:26 +03:00
|
|
|
# frozen_string_literal: true
|
1998-01-16 15:13:05 +03:00
|
|
|
#
|
2003-01-20 18:17:13 +03:00
|
|
|
# find.rb: the Find module for processing all files under a given directory.
|
1998-01-16 15:13:05 +03:00
|
|
|
#
|
|
|
|
|
2003-01-20 18:17:13 +03:00
|
|
|
#
|
|
|
|
# The +Find+ module supports the top-down traversal of a set of file paths.
|
|
|
|
#
|
|
|
|
# For example, to total the size of all files under your home directory,
|
|
|
|
# ignoring anything in a "dot" directory (e.g. $HOME/.ssh):
|
|
|
|
#
|
|
|
|
# require 'find'
|
|
|
|
#
|
|
|
|
# total_size = 0
|
|
|
|
#
|
|
|
|
# Find.find(ENV["HOME"]) do |path|
|
|
|
|
# if FileTest.directory?(path)
|
2019-05-08 05:14:21 +03:00
|
|
|
# if File.basename(path).start_with?('.')
|
2003-01-20 18:17:13 +03:00
|
|
|
# Find.prune # Don't look any further into this directory.
|
|
|
|
# else
|
|
|
|
# next
|
|
|
|
# end
|
|
|
|
# else
|
|
|
|
# total_size += FileTest.size(path)
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
1998-01-16 15:13:05 +03:00
|
|
|
module Find
|
2003-01-20 18:17:13 +03:00
|
|
|
|
2023-11-07 06:44:51 +03:00
|
|
|
VERSION = "0.2.0"
|
2023-03-30 07:05:45 +03:00
|
|
|
|
2003-01-20 18:17:13 +03:00
|
|
|
#
|
|
|
|
# Calls the associated block with the name of every file and directory listed
|
|
|
|
# as arguments, then recursively on their subdirectories, and so on.
|
|
|
|
#
|
2011-09-17 07:49:20 +04:00
|
|
|
# Returns an enumerator if no block is given.
|
|
|
|
#
|
2003-01-20 18:17:13 +03:00
|
|
|
# See the +Find+ module documentation for an example.
|
|
|
|
#
|
2014-03-02 06:15:21 +04:00
|
|
|
def find(*paths, ignore_error: true) # :yield: path
|
2014-03-03 19:28:58 +04:00
|
|
|
block_given? or return enum_for(__method__, *paths, ignore_error: ignore_error)
|
2008-06-10 22:41:25 +04:00
|
|
|
|
2013-09-07 08:34:27 +04:00
|
|
|
fs_encoding = Encoding.find("filesystem")
|
|
|
|
|
2016-02-19 17:38:28 +03:00
|
|
|
paths.collect!{|d| raise Errno::ENOENT, d unless File.exist?(d); d.dup}.each do |path|
|
2014-10-13 11:34:23 +04:00
|
|
|
path = path.to_path if path.respond_to? :to_path
|
2013-09-07 08:34:27 +04:00
|
|
|
enc = path.encoding == Encoding::US_ASCII ? fs_encoding : path.encoding
|
|
|
|
ps = [path]
|
|
|
|
while file = ps.shift
|
|
|
|
catch(:prune) do
|
2019-09-25 06:59:12 +03:00
|
|
|
yield file.dup
|
2009-12-14 15:50:13 +03:00
|
|
|
begin
|
2013-09-07 08:34:27 +04:00
|
|
|
s = File.lstat(file)
|
2021-06-17 19:09:16 +03:00
|
|
|
rescue Errno::ENOENT, Errno::EACCES, Errno::ENOTDIR, Errno::ELOOP, Errno::ENAMETOOLONG, Errno::EINVAL
|
2014-03-02 06:15:21 +04:00
|
|
|
raise unless ignore_error
|
2009-12-14 15:50:13 +03:00
|
|
|
next
|
|
|
|
end
|
2013-09-07 08:34:27 +04:00
|
|
|
if s.directory? then
|
|
|
|
begin
|
2017-09-15 20:00:49 +03:00
|
|
|
fs = Dir.children(file, encoding: enc)
|
2021-06-17 19:09:16 +03:00
|
|
|
rescue Errno::ENOENT, Errno::EACCES, Errno::ENOTDIR, Errno::ELOOP, Errno::ENAMETOOLONG, Errno::EINVAL
|
2014-03-02 06:15:21 +04:00
|
|
|
raise unless ignore_error
|
2013-09-07 08:34:27 +04:00
|
|
|
next
|
|
|
|
end
|
|
|
|
fs.sort!
|
|
|
|
fs.reverse_each {|f|
|
|
|
|
f = File.join(file, f)
|
2019-09-25 06:59:12 +03:00
|
|
|
ps.unshift f
|
2013-09-07 08:34:27 +04:00
|
|
|
}
|
|
|
|
end
|
2009-12-14 15:50:13 +03:00
|
|
|
end
|
2000-10-02 11:48:42 +04:00
|
|
|
end
|
1998-01-16 15:13:05 +03:00
|
|
|
end
|
2013-09-07 08:34:27 +04:00
|
|
|
nil
|
1998-01-16 15:13:05 +03:00
|
|
|
end
|
|
|
|
|
2003-01-20 18:17:13 +03:00
|
|
|
#
|
|
|
|
# Skips the current file or directory, restarting the loop with the next
|
|
|
|
# entry. If the current file is a directory, that directory will not be
|
|
|
|
# recursively entered. Meaningful only within the block associated with
|
|
|
|
# Find::find.
|
|
|
|
#
|
|
|
|
# See the +Find+ module documentation for an example.
|
|
|
|
#
|
1998-01-16 15:13:05 +03:00
|
|
|
def prune
|
|
|
|
throw :prune
|
|
|
|
end
|
2003-01-20 18:17:13 +03:00
|
|
|
|
1998-01-16 15:13:05 +03:00
|
|
|
module_function :find, :prune
|
|
|
|
end
|