2021-06-05 23:08:28 -04:00
|
|
|
# Objects of class Dir are directory streams representing
|
|
|
|
# directories in the underlying file system. They provide a variety
|
|
|
|
# of ways to list directories and their contents. See also File.
|
|
|
|
#
|
|
|
|
# The directory used in these examples contains the two regular files
|
|
|
|
# (<code>config.h</code> and <code>main.rb</code>), the parent
|
|
|
|
# directory (<code>..</code>), and the directory itself
|
|
|
|
# (<code>.</code>).
|
|
|
|
#
|
|
|
|
# == What's Here
|
|
|
|
#
|
2021-09-13 11:37:15 -04:00
|
|
|
# First, what's elsewhere. \Class \Dir:
|
|
|
|
#
|
2022-02-19 01:14:01 -05:00
|
|
|
# - Inherits from {class Object}[rdoc-ref:Object@What-27s+Here].
|
|
|
|
# - Includes {module Enumerable}[rdoc-ref:Enumerable@What-27s+Here],
|
2021-09-13 11:37:15 -04:00
|
|
|
# which provides dozens of additional methods.
|
|
|
|
#
|
|
|
|
# Here, class \Dir provides methods that are useful for:
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
2022-02-19 01:14:01 -05:00
|
|
|
# - {Reading}[rdoc-ref:Dir@Reading]
|
|
|
|
# - {Setting}[rdoc-ref:Dir@Setting]
|
|
|
|
# - {Querying}[rdoc-ref:Dir@Querying]
|
|
|
|
# - {Iterating}[rdoc-ref:Dir@Iterating]
|
|
|
|
# - {Other}[rdoc-ref:Dir@Other]
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
|
|
|
# === Reading
|
|
|
|
#
|
2022-03-30 15:14:36 -04:00
|
|
|
# - #close: Closes the directory stream for +self+.
|
|
|
|
# - #pos=: Sets the position in the directory stream for +self+.
|
|
|
|
# - #read: Reads and returns the next entry in the directory stream for +self+.
|
|
|
|
# - #rewind: Sets the position in the directory stream for +self+ to the first entry.
|
|
|
|
# - #seek: Sets the position in the directory stream for +self+
|
|
|
|
# the entry at the given offset.
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
|
|
|
# === Setting
|
|
|
|
#
|
2022-03-30 15:14:36 -04:00
|
|
|
# - ::chdir: Changes the working directory of the current process
|
|
|
|
# to the given directory.
|
|
|
|
# - ::chroot: Changes the file-system root for the current process
|
|
|
|
# to the given directory.
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
|
|
|
# === Querying
|
|
|
|
#
|
2022-03-30 15:14:36 -04:00
|
|
|
# - ::[]: Same as ::glob without the ability to pass flags.
|
|
|
|
# - ::children: Returns an array of names of the children
|
|
|
|
# (both files and directories) of the given directory,
|
|
|
|
# but not including <tt>.</tt> or <tt>..</tt>.
|
|
|
|
# - ::empty?: Returns whether the given path is an empty directory.
|
|
|
|
# - ::entries: Returns an array of names of the children
|
|
|
|
# (both files and directories) of the given directory,
|
|
|
|
# including <tt>.</tt> and <tt>..</tt>.
|
|
|
|
# - ::exist?: Returns whether the given path is a directory.
|
|
|
|
# - ::getwd (aliased as #pwd): Returns the path to the current working directory.
|
|
|
|
# - ::glob: Returns an array of file paths matching the given pattern and flags.
|
|
|
|
# - ::home: Returns the home directory path for a given user or the current user.
|
|
|
|
# - #children: Returns an array of names of the children
|
|
|
|
# (both files and directories) of +self+,
|
|
|
|
# but not including <tt>.</tt> or <tt>..</tt>.
|
|
|
|
# - #fileno: Returns the integer file descriptor for +self+.
|
|
|
|
# - #path (aliased as #to_path): Returns the path used to create +self+.
|
|
|
|
# - #tell (aliased as #pos): Returns the integer position
|
|
|
|
# in the directory stream for +self+.
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
|
|
|
# === Iterating
|
|
|
|
#
|
2022-03-30 15:14:36 -04:00
|
|
|
# - ::each_child: Calls the given block with each entry in the given directory,
|
|
|
|
# but not including <tt>.</tt> or <tt>..</tt>.
|
|
|
|
# - ::foreach: Calls the given block with each entryin the given directory,
|
|
|
|
# including <tt>.</tt> and <tt>..</tt>.
|
|
|
|
# - #each: Calls the given block with each entry in +self+,
|
|
|
|
# including <tt>.</tt> and <tt>..</tt>.
|
|
|
|
# - #each_child: Calls the given block with each entry in +self+,
|
|
|
|
# but not including <tt>.</tt> or <tt>..</tt>.
|
2021-06-05 23:08:28 -04:00
|
|
|
#
|
|
|
|
# === Other
|
|
|
|
#
|
2022-03-30 15:14:36 -04:00
|
|
|
# - ::mkdir: Creates a directory at the given path, with optional permissions.
|
|
|
|
# - ::new: Returns a new \Dir for the given path, with optional encoding.
|
|
|
|
# - ::open: Same as ::new, but if a block is given, yields the \Dir to the block,
|
|
|
|
# closing it upon block exit.
|
|
|
|
# - ::unlink (aliased as ::delete and ::rmdir): Removes the given directory.
|
|
|
|
# - #inspect: Returns a string description of +self+.
|
2020-01-17 23:59:21 -05:00
|
|
|
class Dir
|
2021-11-04 20:33:05 -04:00
|
|
|
# call-seq:
|
2020-01-17 23:59:21 -05:00
|
|
|
# Dir.open( string ) -> aDir
|
|
|
|
# Dir.open( string, encoding: enc ) -> aDir
|
|
|
|
# Dir.open( string ) {| aDir | block } -> anObject
|
|
|
|
# Dir.open( string, encoding: enc ) {| aDir | block } -> anObject
|
|
|
|
#
|
|
|
|
# The optional <i>encoding</i> keyword argument specifies the encoding of the directory.
|
|
|
|
# If not specified, the filesystem encoding is used.
|
|
|
|
#
|
|
|
|
# With no block, <code>open</code> is a synonym for Dir::new. If a
|
|
|
|
# block is present, it is passed <i>aDir</i> as a parameter. The
|
|
|
|
# directory is closed at the end of the block, and Dir::open returns
|
|
|
|
# the value of the block.
|
|
|
|
def self.open(name, encoding: nil, &block)
|
2020-05-31 02:52:32 -04:00
|
|
|
dir = Primitive.dir_s_open(name, encoding)
|
2020-01-17 23:59:21 -05:00
|
|
|
if block
|
|
|
|
begin
|
|
|
|
yield dir
|
|
|
|
ensure
|
2020-05-31 02:52:32 -04:00
|
|
|
Primitive.dir_s_close(dir)
|
2020-01-17 23:59:21 -05:00
|
|
|
end
|
|
|
|
else
|
|
|
|
dir
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-11-04 20:33:05 -04:00
|
|
|
# call-seq:
|
2020-01-17 23:59:21 -05:00
|
|
|
# Dir.new( string ) -> aDir
|
|
|
|
# Dir.new( string, encoding: enc ) -> aDir
|
|
|
|
#
|
|
|
|
# Returns a new directory object for the named directory.
|
|
|
|
#
|
|
|
|
# The optional <i>encoding</i> keyword argument specifies the encoding of the directory.
|
|
|
|
# If not specified, the filesystem encoding is used.
|
|
|
|
def initialize(name, encoding: nil)
|
2020-05-31 02:52:32 -04:00
|
|
|
Primitive.dir_initialize(name, encoding)
|
2020-01-17 23:59:21 -05:00
|
|
|
end
|
2020-03-24 10:18:36 -04:00
|
|
|
|
2021-11-04 20:33:05 -04:00
|
|
|
# call-seq:
|
2020-03-24 11:53:11 -04:00
|
|
|
# Dir[ string [, string ...] [, base: path] [, sort: true] ] -> array
|
|
|
|
#
|
|
|
|
# Equivalent to calling
|
|
|
|
# <code>Dir.glob([</code><i>string,...</i><code>], 0)</code>.
|
|
|
|
def self.[](*args, base: nil, sort: true)
|
2020-05-31 02:52:32 -04:00
|
|
|
Primitive.dir_s_aref(args, base, sort)
|
2020-03-24 11:53:11 -04:00
|
|
|
end
|
|
|
|
|
2021-11-04 20:33:05 -04:00
|
|
|
# call-seq:
|
2020-03-24 10:18:36 -04:00
|
|
|
# Dir.glob( pattern, [flags], [base: path] [, sort: true] ) -> array
|
|
|
|
# Dir.glob( pattern, [flags], [base: path] [, sort: true] ) { |filename| block } -> nil
|
|
|
|
#
|
|
|
|
# Expands +pattern+, which is a pattern string or an Array of pattern
|
|
|
|
# strings, and returns an array containing the matching filenames.
|
|
|
|
# If a block is given, calls the block once for each matching filename,
|
|
|
|
# passing the filename as a parameter to the block.
|
|
|
|
#
|
|
|
|
# The optional +base+ keyword argument specifies the base directory for
|
|
|
|
# interpreting relative pathnames instead of the current working directory.
|
|
|
|
# As the results are not prefixed with the base directory name in this
|
|
|
|
# case, you will need to prepend the base directory name if you want real
|
|
|
|
# paths.
|
|
|
|
#
|
|
|
|
# The results which matched single wildcard or character set are sorted in
|
2021-11-04 19:51:13 -04:00
|
|
|
# binary ascending order, unless +false+ is given as the optional +sort+
|
2020-03-24 10:18:36 -04:00
|
|
|
# keyword argument. The order of an Array of pattern strings and braces
|
|
|
|
# are preserved.
|
|
|
|
#
|
|
|
|
# Note that the pattern is not a regexp, it's closer to a shell glob.
|
|
|
|
# See File::fnmatch for the meaning of the +flags+ parameter.
|
2021-11-04 19:51:13 -04:00
|
|
|
# Case sensitivity depends on your system (+File::FNM_CASEFOLD+ is ignored).
|
2020-03-24 10:18:36 -04:00
|
|
|
#
|
|
|
|
# <code>*</code>::
|
|
|
|
# Matches any file. Can be restricted by other values in the glob.
|
2021-06-07 07:43:43 -04:00
|
|
|
# Equivalent to <code>/.*/mx</code> in regexp.
|
2020-03-24 10:18:36 -04:00
|
|
|
#
|
|
|
|
# <code>*</code>:: Matches all files
|
|
|
|
# <code>c*</code>:: Matches all files beginning with <code>c</code>
|
|
|
|
# <code>*c</code>:: Matches all files ending with <code>c</code>
|
|
|
|
# <code>\*c\*</code>:: Match all files that have <code>c</code> in them
|
|
|
|
# (including at the beginning or end).
|
|
|
|
#
|
|
|
|
# Note, this will not match Unix-like hidden files (dotfiles). In order
|
|
|
|
# to include those in the match results, you must use the
|
|
|
|
# File::FNM_DOTMATCH flag or something like <code>"{*,.*}"</code>.
|
|
|
|
#
|
|
|
|
# <code>**</code>::
|
2020-10-26 19:51:54 -04:00
|
|
|
# Matches directories recursively if followed by <code>/</code>. If
|
|
|
|
# this path segment contains any other characters, it is the same as the
|
|
|
|
# usual <code>*</code>.
|
2020-03-24 10:18:36 -04:00
|
|
|
#
|
|
|
|
# <code>?</code>::
|
|
|
|
# Matches any one character. Equivalent to <code>/.{1}/</code> in regexp.
|
|
|
|
#
|
|
|
|
# <code>[set]</code>::
|
|
|
|
# Matches any one character in +set+. Behaves exactly like character sets
|
|
|
|
# in Regexp, including set negation (<code>[^a-z]</code>).
|
|
|
|
#
|
|
|
|
# <code>{p,q}</code>::
|
|
|
|
# Matches either literal <code>p</code> or literal <code>q</code>.
|
|
|
|
# Equivalent to pattern alternation in regexp.
|
|
|
|
#
|
|
|
|
# Matching literals may be more than one character in length. More than
|
|
|
|
# two literals may be specified.
|
|
|
|
#
|
2021-06-07 07:43:43 -04:00
|
|
|
# <code>\\</code>::
|
2020-03-24 10:18:36 -04:00
|
|
|
# Escapes the next metacharacter.
|
|
|
|
#
|
|
|
|
# Note that this means you cannot use backslash on windows as part of a
|
|
|
|
# glob, i.e. <code>Dir["c:\\foo*"]</code> will not work, use
|
|
|
|
# <code>Dir["c:/foo*"]</code> instead.
|
|
|
|
#
|
|
|
|
# Examples:
|
|
|
|
#
|
|
|
|
# Dir["config.?"] #=> ["config.h"]
|
|
|
|
# Dir.glob("config.?") #=> ["config.h"]
|
|
|
|
# Dir.glob("*.[a-z][a-z]") #=> ["main.rb"]
|
|
|
|
# Dir.glob("*.[^r]*") #=> ["config.h"]
|
|
|
|
# Dir.glob("*.{rb,h}") #=> ["main.rb", "config.h"]
|
|
|
|
# Dir.glob("*") #=> ["config.h", "main.rb"]
|
2022-01-04 15:38:50 -05:00
|
|
|
# Dir.glob("*", File::FNM_DOTMATCH) #=> [".", "config.h", "main.rb"]
|
2020-03-24 10:18:36 -04:00
|
|
|
# Dir.glob(["*.rb", "*.h"]) #=> ["main.rb", "config.h"]
|
|
|
|
#
|
2020-10-25 20:09:45 -04:00
|
|
|
# Dir.glob("**/*.rb") #=> ["main.rb",
|
2020-03-24 10:18:36 -04:00
|
|
|
# # "lib/song.rb",
|
|
|
|
# # "lib/song/karaoke.rb"]
|
|
|
|
#
|
2020-10-25 20:09:45 -04:00
|
|
|
# Dir.glob("**/*.rb", base: "lib") #=> ["song.rb",
|
2020-03-24 10:18:36 -04:00
|
|
|
# # "song/karaoke.rb"]
|
|
|
|
#
|
2020-10-25 20:09:45 -04:00
|
|
|
# Dir.glob("**/lib") #=> ["lib"]
|
2020-03-24 10:18:36 -04:00
|
|
|
#
|
2020-10-25 20:09:45 -04:00
|
|
|
# Dir.glob("**/lib/**/*.rb") #=> ["lib/song.rb",
|
2020-03-24 10:18:36 -04:00
|
|
|
# # "lib/song/karaoke.rb"]
|
|
|
|
#
|
2020-10-25 20:09:45 -04:00
|
|
|
# Dir.glob("**/lib/*.rb") #=> ["lib/song.rb"]
|
2020-03-24 10:18:36 -04:00
|
|
|
def self.glob(pattern, _flags = 0, flags: _flags, base: nil, sort: true)
|
2020-05-31 02:52:32 -04:00
|
|
|
Primitive.dir_s_glob(pattern, flags, base, sort)
|
2020-03-24 10:18:36 -04:00
|
|
|
end
|
2020-01-17 23:59:21 -05:00
|
|
|
end
|
2021-05-20 09:57:33 -04:00
|
|
|
|
|
|
|
class << File
|
|
|
|
# call-seq:
|
|
|
|
# File.fnmatch( pattern, path, [flags] ) -> (true or false)
|
|
|
|
# File.fnmatch?( pattern, path, [flags] ) -> (true or false)
|
|
|
|
#
|
|
|
|
# Returns true if +path+ matches against +pattern+. The pattern is not a
|
|
|
|
# regular expression; instead it follows rules similar to shell filename
|
|
|
|
# globbing. It may contain the following metacharacters:
|
|
|
|
#
|
|
|
|
# <code>*</code>::
|
|
|
|
# Matches any file. Can be restricted by other values in the glob.
|
2021-06-07 07:43:43 -04:00
|
|
|
# Equivalent to <code>/.*/x</code> in regexp.
|
2021-05-20 09:57:33 -04:00
|
|
|
#
|
2021-08-26 07:44:34 -04:00
|
|
|
# <code>*</code>:: Matches all regular files
|
2021-05-20 09:57:33 -04:00
|
|
|
# <code>c*</code>:: Matches all files beginning with <code>c</code>
|
|
|
|
# <code>*c</code>:: Matches all files ending with <code>c</code>
|
|
|
|
# <code>\*c*</code>:: Matches all files that have <code>c</code> in them
|
|
|
|
# (including at the beginning or end).
|
|
|
|
#
|
2021-08-31 08:02:13 -04:00
|
|
|
# To match hidden files (that start with a <code>.</code>) set the
|
2021-05-20 09:57:33 -04:00
|
|
|
# File::FNM_DOTMATCH flag.
|
|
|
|
#
|
|
|
|
# <code>**</code>::
|
|
|
|
# Matches directories recursively or files expansively.
|
|
|
|
#
|
|
|
|
# <code>?</code>::
|
|
|
|
# Matches any one character. Equivalent to <code>/.{1}/</code> in regexp.
|
|
|
|
#
|
|
|
|
# <code>[set]</code>::
|
|
|
|
# Matches any one character in +set+. Behaves exactly like character sets
|
|
|
|
# in Regexp, including set negation (<code>[^a-z]</code>).
|
|
|
|
#
|
2021-06-07 07:43:43 -04:00
|
|
|
# <code>\\</code>::
|
2021-05-20 09:57:33 -04:00
|
|
|
# Escapes the next metacharacter.
|
|
|
|
#
|
|
|
|
# <code>{a,b}</code>::
|
|
|
|
# Matches pattern a and pattern b if File::FNM_EXTGLOB flag is enabled.
|
|
|
|
# Behaves like a Regexp union (<code>(?:a|b)</code>).
|
|
|
|
#
|
|
|
|
# +flags+ is a bitwise OR of the <code>FNM_XXX</code> constants. The same
|
|
|
|
# glob pattern and flags are used by Dir::glob.
|
|
|
|
#
|
|
|
|
# Examples:
|
|
|
|
#
|
|
|
|
# File.fnmatch('cat', 'cat') #=> true # match entire string
|
|
|
|
# File.fnmatch('cat', 'category') #=> false # only match partial string
|
|
|
|
#
|
|
|
|
# File.fnmatch('c{at,ub}s', 'cats') #=> false # { } isn't supported by default
|
|
|
|
# File.fnmatch('c{at,ub}s', 'cats', File::FNM_EXTGLOB) #=> true # { } is supported on FNM_EXTGLOB
|
|
|
|
#
|
|
|
|
# File.fnmatch('c?t', 'cat') #=> true # '?' match only 1 character
|
|
|
|
# File.fnmatch('c??t', 'cat') #=> false # ditto
|
|
|
|
# File.fnmatch('c*', 'cats') #=> true # '*' match 0 or more characters
|
|
|
|
# File.fnmatch('c*t', 'c/a/b/t') #=> true # ditto
|
|
|
|
# File.fnmatch('ca[a-z]', 'cat') #=> true # inclusive bracket expression
|
|
|
|
# File.fnmatch('ca[^t]', 'cat') #=> false # exclusive bracket expression ('^' or '!')
|
|
|
|
#
|
|
|
|
# File.fnmatch('cat', 'CAT') #=> false # case sensitive
|
|
|
|
# File.fnmatch('cat', 'CAT', File::FNM_CASEFOLD) #=> true # case insensitive
|
|
|
|
# File.fnmatch('cat', 'CAT', File::FNM_SYSCASE) #=> true or false # depends on the system default
|
|
|
|
#
|
|
|
|
# File.fnmatch('?', '/', File::FNM_PATHNAME) #=> false # wildcard doesn't match '/' on FNM_PATHNAME
|
|
|
|
# File.fnmatch('*', '/', File::FNM_PATHNAME) #=> false # ditto
|
|
|
|
# File.fnmatch('[/]', '/', File::FNM_PATHNAME) #=> false # ditto
|
|
|
|
#
|
|
|
|
# File.fnmatch('\?', '?') #=> true # escaped wildcard becomes ordinary
|
|
|
|
# File.fnmatch('\a', 'a') #=> true # escaped ordinary remains ordinary
|
|
|
|
# File.fnmatch('\a', '\a', File::FNM_NOESCAPE) #=> true # FNM_NOESCAPE makes '\' ordinary
|
|
|
|
# File.fnmatch('[\?]', '?') #=> true # can escape inside bracket expression
|
|
|
|
#
|
|
|
|
# File.fnmatch('*', '.profile') #=> false # wildcard doesn't match leading
|
|
|
|
# File.fnmatch('*', '.profile', File::FNM_DOTMATCH) #=> true # period by default.
|
|
|
|
# File.fnmatch('.*', '.profile') #=> true
|
|
|
|
#
|
2020-10-26 11:52:43 -04:00
|
|
|
# File.fnmatch('**/*.rb', 'main.rb') #=> false
|
|
|
|
# File.fnmatch('**/*.rb', './main.rb') #=> false
|
|
|
|
# File.fnmatch('**/*.rb', 'lib/song.rb') #=> true
|
2021-05-20 09:57:33 -04:00
|
|
|
# File.fnmatch('**.rb', 'main.rb') #=> true
|
|
|
|
# File.fnmatch('**.rb', './main.rb') #=> false
|
|
|
|
# File.fnmatch('**.rb', 'lib/song.rb') #=> true
|
|
|
|
# File.fnmatch('*', 'dave/.profile') #=> true
|
|
|
|
#
|
2020-10-26 11:52:43 -04:00
|
|
|
# File.fnmatch('**/foo', 'a/b/c/foo', File::FNM_PATHNAME) #=> true
|
|
|
|
# File.fnmatch('**/foo', '/a/b/c/foo', File::FNM_PATHNAME) #=> true
|
|
|
|
# File.fnmatch('**/foo', 'c:/a/b/c/foo', File::FNM_PATHNAME) #=> true
|
|
|
|
# File.fnmatch('**/foo', 'a/.b/c/foo', File::FNM_PATHNAME) #=> false
|
|
|
|
# File.fnmatch('**/foo', 'a/.b/c/foo', File::FNM_PATHNAME | File::FNM_DOTMATCH) #=> true
|
2021-07-01 01:26:06 -04:00
|
|
|
def fnmatch(pattern, path, flags = 0)
|
2021-05-20 09:57:33 -04:00
|
|
|
end
|
|
|
|
alias fnmatch? fnmatch
|
|
|
|
end if false
|