2017-01-07 05:14:07 +03:00
# frozen_string_literal: true
2018-10-03 22:19:33 +03:00
begin
require 'rbconfig'
rescue LoadError
2023-03-07 10:17:25 +03:00
# for make rjit-headers
2018-10-03 22:19:33 +03:00
end
2022-06-17 16:36:20 +03:00
# Namespace for file utility methods for copying, moving, removing, etc.
2018-10-20 10:03:55 +03:00
#
2022-06-17 16:36:20 +03:00
# == What's Here
2009-03-06 06:56:38 +03:00
#
2022-06-17 16:36:20 +03:00
# First, what’ s elsewhere. \Module \FileUtils:
2009-03-06 06:56:38 +03:00
#
2022-10-12 05:54:16 +03:00
# - Inherits from {class Object}[rdoc-ref:Object].
# - Supplements {class File}[rdoc-ref:File]
2022-06-17 16:36:20 +03:00
# (but is not included or extended there).
2009-03-06 06:56:38 +03:00
#
2022-06-17 16:36:20 +03:00
# Here, module \FileUtils provides methods that are useful for:
2009-03-06 06:56:38 +03:00
#
2022-06-17 16:36:20 +03:00
# - {Creating}[rdoc-ref:FileUtils@Creating].
# - {Deleting}[rdoc-ref:FileUtils@Deleting].
# - {Querying}[rdoc-ref:FileUtils@Querying].
# - {Setting}[rdoc-ref:FileUtils@Setting].
# - {Comparing}[rdoc-ref:FileUtils@Comparing].
# - {Copying}[rdoc-ref:FileUtils@Copying].
2022-06-21 20:23:06 +03:00
# - {Moving}[rdoc-ref:FileUtils@Moving].
2022-06-17 16:36:20 +03:00
# - {Options}[rdoc-ref:FileUtils@Options].
#
# === Creating
#
# - ::mkdir: Creates directories.
# - ::mkdir_p, ::makedirs, ::mkpath: Creates directories,
# also creating ancestor directories as needed.
# - ::link_entry: Creates a hard link.
# - ::ln, ::link: Creates hard links.
# - ::ln_s, ::symlink: Creates symbolic links.
# - ::ln_sf: Creates symbolic links, overwriting if necessary.
2022-11-25 04:03:51 +03:00
# - ::ln_sr: Creates symbolic links relative to targets
2022-06-17 16:36:20 +03:00
#
# === Deleting
#
# - ::remove_dir: Removes a directory and its descendants.
# - ::remove_entry: Removes an entry, including its descendants if it is a directory.
# - ::remove_entry_secure: Like ::remove_entry, but removes securely.
# - ::remove_file: Removes a file entry.
# - ::rm, ::remove: Removes entries.
# - ::rm_f, ::safe_unlink: Like ::rm, but removes forcibly.
# - ::rm_r: Removes entries and their descendants.
# - ::rm_rf, ::rmtree: Like ::rm_r, but removes forcibly.
# - ::rmdir: Removes directories.
#
# === Querying
#
# - ::pwd, ::getwd: Returns the path to the working directory.
# - ::uptodate?: Returns whether a given entry is newer than given other entries.
#
# === Setting
#
# - ::cd, ::chdir: Sets the working directory.
# - ::chmod: Sets permissions for an entry.
# - ::chmod_R: Sets permissions for an entry and its descendants.
# - ::chown: Sets the owner and group for entries.
# - ::chown_R: Sets the owner and group for entries and their descendants.
# - ::touch: Sets modification and access times for entries,
# creating if necessary.
#
# === Comparing
#
# - ::compare_file, ::cmp, ::identical?: Returns whether two entries are identical.
# - ::compare_stream: Returns whether two streams are identical.
#
# === Copying
#
# - ::copy_entry: Recursively copies an entry.
# - ::copy_file: Copies an entry.
# - ::copy_stream: Copies a stream.
# - ::cp, ::copy: Copies files.
# - ::cp_lr: Recursively creates hard links.
2022-06-21 20:23:06 +03:00
# - ::cp_r: Recursively copies files, retaining mode, owner, and group.
# - ::install: Recursively copies files, optionally setting mode,
# owner, and group.
2022-06-17 16:36:20 +03:00
#
# === Moving
#
# - ::mv, ::move: Moves entries.
#
# === Options
#
# - ::collect_method: Returns the names of methods that accept a given option.
# - ::commands: Returns the names of methods that accept options.
# - ::have_option?: Returns whether a given method accepts a given option.
# - ::options: Returns all option names.
# - ::options_of: Returns the names of the options for a given method.
2009-03-06 06:56:38 +03:00
#
2022-06-13 18:39:41 +03:00
# == Path Arguments
#
# Some methods in \FileUtils accept _path_ arguments,
# which are interpreted as paths to filesystem entries:
#
# - If the argument is a string, that value is the path.
# - If the argument has method +:to_path+, it is converted via that method.
# - If the argument has method +:to_str+, it is converted via that method.
#
2022-06-08 15:34:47 +03:00
# == About the Examples
#
# Some examples here involve trees of file entries.
# For these, we sometimes display trees using the
# {tree command-line utility}[https://en.wikipedia.org/wiki/Tree_(command)],
# which is a recursive directory-listing utility that produces
# a depth-indented listing of files and directories.
#
# We use a helper method to launch the command and control the format:
#
# def tree(dirpath = '.')
# command = "tree --noreport --charset=ascii #{dirpath}"
# system(command)
# end
#
2022-06-28 18:39:44 +03:00
# To illustrate:
#
# tree('src0')
# # => src0
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
2022-06-08 15:34:47 +03:00
#
2022-06-06 18:37:18 +03:00
# == Avoiding the TOCTTOU Vulnerability
#
# For certain methods that recursively remove entries,
# there is a potential vulnerability called the
# {Time-of-check to time-of-use}[https://en.wikipedia.org/wiki/Time-of-check_to_time-of-use],
# or TOCTTOU, vulnerability that can exist when:
#
# - An ancestor directory of the entry at the target path is world writable;
# such directories include <tt>/tmp</tt>.
# - The directory tree at the target path includes:
#
# - A world-writable descendant directory.
# - A symbolic link.
#
# To avoid that vulnerability, you can use this method to remove entries:
#
# - FileUtils.remove_entry_secure: removes recursively
# if the target path points to a directory.
#
# Also available are these methods,
# each of which calls \FileUtils.remove_entry_secure:
#
# - FileUtils.rm_r with keyword argument <tt>secure: true</tt>.
# - FileUtils.rm_rf with keyword argument <tt>secure: true</tt>.
#
# Finally, this method for moving entries calls \FileUtils.remove_entry_secure
2022-06-15 23:56:36 +03:00
# if the source and destination are on different file systems
2022-06-06 18:37:18 +03:00
# (which means that the "move" is really a copy and remove):
#
# - FileUtils.mv with keyword argument <tt>secure: true</tt>.
#
# \Method \FileUtils.remove_entry_secure removes securely
# by applying a special pre-process:
#
2022-06-21 17:16:20 +03:00
# - If the target path points to a directory, this method uses methods
2022-10-12 05:54:16 +03:00
# {File#chown}[rdoc-ref:File#chown]
# and {File#chmod}[rdoc-ref:File#chmod]
2022-06-06 18:37:18 +03:00
# in removing directories.
# - The owner of the target directory should be either the current process
# or the super user (root).
#
# WARNING: You must ensure that *ALL* parent directories cannot be
# moved by other untrusted users. For example, parent directories
# should not be owned by untrusted users, and should not be world
# writable except when the sticky bit is set.
#
# For details of this security vulnerability, see Perl cases:
#
# - {CVE-2005-0448}[https://cve.mitre.org/cgi-bin/cvename.cgi?name=CAN-2005-0448].
# - {CVE-2004-0452}[https://cve.mitre.org/cgi-bin/cvename.cgi?name=CAN-2004-0452].
#
2002-03-26 05:03:04 +03:00
module FileUtils
2024-02-21 19:52:59 +03:00
# The version number.
2023-11-07 10:15:19 +03:00
VERSION = " 1.7.2 "
2002-03-26 05:03:04 +03:00
2013-03-01 06:09:42 +04:00
def self . private_module_function ( name ) #:nodoc:
module_function name
private_class_method name
2005-09-19 00:59:29 +04:00
end
2003-01-24 19:13:33 +03:00
2003-02-09 01:01:53 +03:00
#
2022-05-10 20:48:53 +03:00
# Returns a string containing the path to the current directory:
#
# FileUtils.pwd # => "/rdoc/fileutils"
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.cd.
#
2003-02-09 01:01:53 +03:00
def pwd
Dir . pwd
end
2013-03-01 06:09:42 +04:00
module_function :pwd
2003-02-09 01:01:53 +03:00
alias getwd pwd
2013-03-01 06:09:42 +04:00
module_function :getwd
2003-02-09 01:01:53 +03:00
2022-06-13 18:39:41 +03:00
# Changes the working directory to the given +dir+, which
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments]:
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# With no block given,
2022-06-13 18:39:41 +03:00
# changes the current directory to the directory at +dir+; returns zero:
2022-05-10 20:48:53 +03:00
#
# FileUtils.pwd # => "/rdoc/fileutils"
# FileUtils.cd('..')
# FileUtils.pwd # => "/rdoc"
# FileUtils.cd('fileutils')
#
# With a block given, changes the current directory to the directory
2022-06-13 18:39:41 +03:00
# at +dir+, calls the block with argument +dir+,
2022-05-10 20:48:53 +03:00
# and restores the original current directory; returns the block's value:
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# FileUtils.pwd # => "/rdoc/fileutils"
# FileUtils.cd('..') { |arg| [arg, FileUtils.pwd] } # => ["..", "/rdoc"]
# FileUtils.pwd # => "/rdoc/fileutils"
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# Keyword arguments:
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# - <tt>verbose: true</tt> - prints an equivalent command:
2019-03-27 11:26:09 +03:00
#
2022-05-10 20:48:53 +03:00
# FileUtils.cd('..')
# FileUtils.cd('fileutils')
#
# Output:
#
# cd ..
# cd fileutils
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.pwd.
#
2016-03-01 05:26:44 +03:00
def cd ( dir , verbose : nil , & block ) # :yield: dir
fu_output_message " cd #{ dir } " if verbose
2018-05-15 08:53:18 +03:00
result = Dir . chdir ( dir , & block )
2016-03-01 05:26:44 +03:00
fu_output_message 'cd -' if verbose and block
2018-05-15 08:53:18 +03:00
result
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :cd
2002-03-26 05:03:04 +03:00
alias chdir cd
2013-03-01 06:09:42 +04:00
module_function :chdir
2002-03-26 05:03:04 +03:00
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# Returns +true+ if the file at path +new+
2022-05-11 00:13:26 +03:00
# is newer than all the files at paths in array +old_list+;
2022-06-13 18:39:41 +03:00
# +false+ otherwise.
#
# Argument +new+ and the elements of +old_list+
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments]:
2022-05-10 20:48:53 +03:00
#
# FileUtils.uptodate?('Rakefile', ['Gemfile', 'README.md']) # => true
# FileUtils.uptodate?('Gemfile', ['Rakefile', 'README.md']) # => false
2009-03-06 06:56:38 +03:00
#
2022-05-10 20:48:53 +03:00
# A non-existent file is considered to be infinitely old.
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.touch.
#
2012-07-16 12:15:05 +04:00
def uptodate? ( new , old_list )
2003-11-18 08:09:20 +03:00
return false unless File . exist? ( new )
2003-02-10 13:48:38 +03:00
new_time = File . mtime ( new )
2002-12-27 07:21:27 +03:00
old_list . each do | old |
2003-11-18 08:09:20 +03:00
if File . exist? ( old )
2002-03-26 05:03:04 +03:00
return false unless new_time > File . mtime ( old )
end
end
true
end
2013-03-01 06:09:42 +04:00
module_function :uptodate?
2002-03-26 05:03:04 +03:00
2017-03-01 13:17:42 +03:00
def remove_trailing_slash ( dir ) #:nodoc:
2013-06-03 18:20:15 +04:00
dir == '/' ? dir : dir . chomp ( ?/ )
end
2015-10-26 11:30:20 +03:00
private_module_function :remove_trailing_slash
2013-06-03 18:20:15 +04:00
2009-03-06 06:56:38 +03:00
#
2022-05-11 00:13:26 +03:00
# Creates directories at the paths in the given +list+
2022-06-13 18:39:41 +03:00
# (a single path or an array of paths);
2022-06-06 18:37:18 +03:00
# returns +list+ if it is an array, <tt>[list]</tt> otherwise.
2022-05-10 20:48:53 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-10 20:48:53 +03:00
# With no keyword arguments, creates a directory at each +path+ in +list+
# by calling: <tt>Dir.mkdir(path, mode)</tt>;
2022-10-12 05:54:16 +03:00
# see {Dir.mkdir}[rdoc-ref:Dir.mkdir]:
2022-05-10 20:48:53 +03:00
#
# FileUtils.mkdir(%w[tmp0 tmp1]) # => ["tmp0", "tmp1"]
2022-05-11 00:13:26 +03:00
# FileUtils.mkdir('tmp4') # => ["tmp4"]
2022-05-10 20:48:53 +03:00
#
# Keyword arguments:
#
2022-06-08 23:37:57 +03:00
# - <tt>mode: <i>mode</i></tt> - also calls <tt>File.chmod(mode, path)</tt>;
2022-10-12 05:54:16 +03:00
# see {File.chmod}[rdoc-ref:File.chmod].
2022-05-10 20:48:53 +03:00
# - <tt>noop: true</tt> - does not create directories.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.mkdir(%w[tmp0 tmp1], verbose: true)
# FileUtils.mkdir(%w[tmp2 tmp3], mode: 0700, verbose: true)
#
# Output:
#
# mkdir tmp0 tmp1
# mkdir -m 700 tmp2 tmp3
2009-03-06 06:56:38 +03:00
#
2022-06-13 18:39:41 +03:00
# Raises an exception if any path points to an existing
2022-05-10 20:48:53 +03:00
# file or directory, or if for any reason a directory cannot be created.
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.mkdir_p.
#
2016-03-01 05:26:44 +03:00
def mkdir ( list , mode : nil , noop : nil , verbose : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message " mkdir #{ mode ? ( '-m %03o ' % mode ) : '' } #{ list . join ' ' } " if verbose
return if noop
2002-03-26 05:03:04 +03:00
list . each do | dir |
2016-03-01 05:26:44 +03:00
fu_mkdir dir , mode
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
module_function :mkdir
2002-03-26 05:03:04 +03:00
2009-03-06 06:56:38 +03:00
#
2022-05-11 18:48:19 +03:00
# Creates directories at the paths in the given +list+
2022-06-14 16:52:18 +03:00
# (a single path or an array of paths),
2022-05-11 18:48:19 +03:00
# also creating ancestor directories as needed;
2022-06-06 18:37:18 +03:00
# returns +list+ if it is an array, <tt>[list]</tt> otherwise.
2022-05-11 18:48:19 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-11 18:48:19 +03:00
# With no keyword arguments, creates a directory at each +path+ in +list+,
# along with any needed ancestor directories,
# by calling: <tt>Dir.mkdir(path, mode)</tt>;
2022-10-12 05:54:16 +03:00
# see {Dir.mkdir}[rdoc-ref:Dir.mkdir]:
2022-05-11 18:48:19 +03:00
#
# FileUtils.mkdir_p(%w[tmp0/tmp1 tmp2/tmp3]) # => ["tmp0/tmp1", "tmp2/tmp3"]
# FileUtils.mkdir_p('tmp4/tmp5') # => ["tmp4/tmp5"]
#
# Keyword arguments:
#
2022-06-08 23:37:57 +03:00
# - <tt>mode: <i>mode</i></tt> - also calls <tt>File.chmod(mode, path)</tt>;
2022-10-12 05:54:16 +03:00
# see {File.chmod}[rdoc-ref:File.chmod].
2022-05-11 18:48:19 +03:00
# - <tt>noop: true</tt> - does not create directories.
# - <tt>verbose: true</tt> - prints an equivalent command:
2009-03-06 06:56:38 +03:00
#
2022-05-11 18:48:19 +03:00
# FileUtils.mkdir_p(%w[tmp0 tmp1], verbose: true)
# FileUtils.mkdir_p(%w[tmp2 tmp3], mode: 0700, verbose: true)
2009-03-06 06:56:38 +03:00
#
2022-05-11 18:48:19 +03:00
# Output:
2017-03-01 13:17:42 +03:00
#
2022-05-11 18:48:19 +03:00
# mkdir -p tmp0 tmp1
# mkdir -p -m 700 tmp2 tmp3
2003-01-24 19:13:33 +03:00
#
2022-05-11 18:48:19 +03:00
# Raises an exception if for any reason a directory cannot be created.
2009-03-06 06:56:38 +03:00
#
2022-06-16 17:56:33 +03:00
# FileUtils.mkpath and FileUtils.makedirs are aliases for FileUtils.mkdir_p.
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.mkdir.
#
2016-03-01 05:26:44 +03:00
def mkdir_p ( list , mode : nil , noop : nil , verbose : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message " mkdir -p #{ mode ? ( '-m %03o ' % mode ) : '' } #{ list . join ' ' } " if verbose
return * list if noop
2002-03-26 05:03:04 +03:00
2020-09-10 20:58:35 +03:00
list . each do | item |
path = remove_trailing_slash ( item )
2002-03-26 05:03:04 +03:00
stack = [ ]
2022-08-01 21:35:37 +03:00
until File . directory? ( path ) || File . dirname ( path ) == path
2003-12-11 13:15:54 +03:00
stack . push path
path = File . dirname ( path )
2002-03-26 05:03:04 +03:00
end
2006-10-09 18:41:24 +04:00
stack . reverse_each do | dir |
2003-12-11 13:15:54 +03:00
begin
2016-03-01 05:26:44 +03:00
fu_mkdir dir , mode
2010-06-22 09:46:01 +04:00
rescue SystemCallError
2006-10-09 18:41:24 +04:00
raise unless File . directory? ( dir )
2003-12-11 13:15:54 +03:00
end
2002-03-26 05:03:04 +03:00
end
end
return * list
end
2013-03-01 06:09:42 +04:00
module_function :mkdir_p
2002-03-26 05:03:04 +03:00
alias mkpath mkdir_p
alias makedirs mkdir_p
2013-03-01 06:09:42 +04:00
module_function :mkpath
module_function :makedirs
2002-03-26 05:03:04 +03:00
2005-09-19 00:59:29 +04:00
def fu_mkdir ( path , mode ) #:nodoc:
2015-10-26 11:30:20 +03:00
path = remove_trailing_slash ( path )
2004-12-04 14:46:24 +03:00
if mode
Dir . mkdir path , mode
File . chmod mode , path
else
Dir . mkdir path
end
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_mkdir
2004-12-04 14:46:24 +03:00
2009-03-06 06:56:38 +03:00
#
2022-05-11 18:48:19 +03:00
# Removes directories at the paths in the given +list+
2022-06-14 16:52:18 +03:00
# (a single path or an array of paths);
2022-06-06 18:37:18 +03:00
# returns +list+, if it is an array, <tt>[list]</tt> otherwise.
2022-05-11 18:48:19 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-11 18:48:19 +03:00
# With no keyword arguments, removes the directory at each +path+ in +list+,
2022-05-11 21:33:30 +03:00
# by calling: <tt>Dir.rmdir(path)</tt>;
2022-10-12 05:54:16 +03:00
# see {Dir.rmdir}[rdoc-ref:Dir.rmdir]:
2022-05-11 18:48:19 +03:00
#
# FileUtils.rmdir(%w[tmp0/tmp1 tmp2/tmp3]) # => ["tmp0/tmp1", "tmp2/tmp3"]
# FileUtils.rmdir('tmp4/tmp5') # => ["tmp4/tmp5"]
#
# Keyword arguments:
#
# - <tt>parents: true</tt> - removes successive ancestor directories
# if empty.
# - <tt>noop: true</tt> - does not remove directories.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.rmdir(%w[tmp0/tmp1 tmp2/tmp3], parents: true, verbose: true)
# FileUtils.rmdir('tmp4/tmp5', parents: true, verbose: true)
#
# Output:
#
# rmdir -p tmp0/tmp1 tmp2/tmp3
# rmdir -p tmp4/tmp5
2009-03-06 06:56:38 +03:00
#
2022-05-11 18:48:19 +03:00
# Raises an exception if a directory does not exist
# or if for any reason a directory cannot be removed.
2009-03-06 06:56:38 +03:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2016-03-01 05:26:44 +03:00
def rmdir ( list , parents : nil , noop : nil , verbose : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message " rmdir #{ parents ? '-p ' : '' } #{ list . join ' ' } " if verbose
return if noop
2002-03-26 05:03:04 +03:00
list . each do | dir |
2017-09-16 14:59:47 +03:00
Dir . rmdir ( dir = remove_trailing_slash ( dir ) )
if parents
begin
2009-01-25 05:06:29 +03:00
until ( parent = File . dirname ( dir ) ) == '.' or parent == dir
2013-06-13 17:12:35 +04:00
dir = parent
2009-01-25 05:06:29 +03:00
Dir . rmdir ( dir )
end
2017-09-16 14:59:47 +03:00
rescue Errno :: ENOTEMPTY , Errno :: EEXIST , Errno :: ENOENT
2009-01-25 05:06:29 +03:00
end
end
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
module_function :rmdir
2002-03-26 05:03:04 +03:00
2022-05-21 02:55:43 +03:00
# Creates {hard links}[https://en.wikipedia.org/wiki/Hard_link].
2003-01-24 19:13:33 +03:00
#
2022-06-14 16:52:18 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-14 16:38:09 +03:00
# When +src+ is the path to an existing file
# and +dest+ is the path to a non-existent file,
# creates a hard link at +dest+ pointing to +src+; returns zero:
#
# Dir.children('tmp0/') # => ["t.txt"]
# Dir.children('tmp1/') # => []
# FileUtils.ln('tmp0/t.txt', 'tmp1/t.lnk') # => 0
# Dir.children('tmp1/') # => ["t.lnk"]
#
# When +src+ is the path to an existing file
# and +dest+ is the path to an existing directory,
2022-05-21 02:55:43 +03:00
# creates a hard link at <tt>dest/src</tt> pointing to +src+; returns zero:
2022-05-14 16:38:09 +03:00
#
# Dir.children('tmp2') # => ["t.dat"]
# Dir.children('tmp3') # => []
# FileUtils.ln('tmp2/t.dat', 'tmp3') # => 0
# Dir.children('tmp3') # => ["t.dat"]
#
# When +src+ is an array of paths to existing files
# and +dest+ is the path to an existing directory,
# then for each path +target+ in +src+,
2022-05-21 02:55:43 +03:00
# creates a hard link at <tt>dest/target</tt> pointing to +target+;
2022-05-14 16:38:09 +03:00
# returns +src+:
#
# Dir.children('tmp4/') # => []
# FileUtils.ln(['tmp0/t.txt', 'tmp2/t.dat'], 'tmp4/') # => ["tmp0/t.txt", "tmp2/t.dat"]
# Dir.children('tmp4/') # => ["t.dat", "t.txt"]
2003-01-24 19:13:33 +03:00
#
2022-05-14 16:38:09 +03:00
# Keyword arguments:
#
# - <tt>force: true</tt> - overwrites +dest+ if it exists.
# - <tt>noop: true</tt> - does not create links.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.ln('tmp0/t.txt', 'tmp1/t.lnk', verbose: true)
# FileUtils.ln('tmp2/t.dat', 'tmp3', verbose: true)
# FileUtils.ln(['tmp0/t.txt', 'tmp2/t.dat'], 'tmp4/', verbose: true)
#
# Output:
2009-03-06 06:56:38 +03:00
#
2022-05-14 16:38:09 +03:00
# ln tmp0/t.txt tmp1/t.lnk
# ln tmp2/t.dat tmp3
# ln tmp0/t.txt tmp2/t.dat tmp4/
2009-03-06 06:56:38 +03:00
#
2022-05-14 16:38:09 +03:00
# Raises an exception if +dest+ is the path to an existing file
# and keyword argument +force+ is not +true+.
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.link_entry (has different options).
#
2016-03-01 05:26:44 +03:00
def ln ( src , dest , force : nil , noop : nil , verbose : nil )
fu_output_message " ln #{ force ? ' -f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
2003-11-18 08:09:20 +03:00
fu_each_src_dest0 ( src , dest ) do | s , d |
2016-03-01 05:26:44 +03:00
remove_file d , true if force
2002-03-26 05:03:04 +03:00
File . link s , d
end
end
2013-03-01 06:09:42 +04:00
module_function :ln
2002-03-26 05:03:04 +03:00
alias link ln
2013-03-01 06:09:42 +04:00
module_function :link
2002-03-26 05:03:04 +03:00
2022-05-21 02:55:43 +03:00
# Creates {hard links}[https://en.wikipedia.org/wiki/Hard_link].
#
2022-06-14 16:52:18 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-21 02:55:43 +03:00
# If +src+ is the path to a directory and +dest+ does not exist,
# creates links +dest+ and descendents pointing to +src+ and its descendents:
#
2022-06-28 18:39:44 +03:00
# tree('src0')
# # => src0
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
# File.exist?('dest0') # => false
# FileUtils.cp_lr('src0', 'dest0')
# tree('dest0')
# # => dest0
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
2022-05-21 02:55:43 +03:00
#
# If +src+ and +dest+ are both paths to directories,
# creates links <tt>dest/src</tt> and descendents
# pointing to +src+ and its descendents:
#
2022-06-28 18:39:44 +03:00
# tree('src1')
# # => src1
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
# FileUtils.mkdir('dest1')
# FileUtils.cp_lr('src1', 'dest1')
# tree('dest1')
# # => dest1
# # `-- src1
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
#
# If +src+ is an array of paths to entries and +dest+ is the path to a directory,
2022-06-15 23:56:36 +03:00
# for each path +filepath+ in +src+, creates a link at <tt>dest/filepath</tt>
# pointing to that path:
#
2022-06-28 18:39:44 +03:00
# tree('src2')
# # => src2
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
# FileUtils.mkdir('dest2')
# FileUtils.cp_lr(['src2/sub0', 'src2/sub1'], 'dest2')
# tree('dest2')
# # => dest2
# # |-- sub0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- sub1
# # |-- src2.txt
# # `-- src3.txt
2022-06-15 23:56:36 +03:00
#
2022-05-21 02:55:43 +03:00
# Keyword arguments:
2018-03-13 11:18:03 +03:00
#
2022-05-23 19:17:07 +03:00
# - <tt>dereference_root: false</tt> - if +src+ is a symbolic link,
# does not dereference it.
2022-05-21 02:55:43 +03:00
# - <tt>noop: true</tt> - does not create links.
# - <tt>remove_destination: true</tt> - removes +dest+ before creating links.
# - <tt>verbose: true</tt> - prints an equivalent command:
2019-03-10 14:09:59 +03:00
#
2022-06-28 18:39:44 +03:00
# FileUtils.cp_lr('src0', 'dest0', noop: true, verbose: true)
# FileUtils.cp_lr('src1', 'dest1', noop: true, verbose: true)
# FileUtils.cp_lr(['src2/sub0', 'src2/sub1'], 'dest2', noop: true, verbose: true)
2019-03-10 14:09:59 +03:00
#
2022-05-21 02:55:43 +03:00
# Output:
2018-03-13 11:18:03 +03:00
#
2022-06-28 18:39:44 +03:00
# cp -lr src0 dest0
# cp -lr src1 dest1
# cp -lr src2/sub0 src2/sub1 dest2
2018-03-13 11:18:03 +03:00
#
2022-05-23 19:17:07 +03:00
# Raises an exception if +dest+ is the path to an existing file or directory
# and keyword argument <tt>remove_destination: true</tt> is not given.
2018-03-13 11:18:03 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2018-03-13 11:18:03 +03:00
def cp_lr ( src , dest , noop : nil , verbose : nil ,
dereference_root : true , remove_destination : false )
fu_output_message " cp -lr #{ remove_destination ? ' --remove-destination' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
fu_each_src_dest ( src , dest ) do | s , d |
link_entry s , d , dereference_root , remove_destination
end
end
module_function :cp_lr
2022-05-23 16:00:21 +03:00
# Creates {symbolic links}[https://en.wikipedia.org/wiki/Symbolic_link].
2003-01-24 19:13:33 +03:00
#
2022-06-14 16:52:18 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-15 23:56:36 +03:00
# If +src+ is the path to an existing file:
2022-05-23 16:00:21 +03:00
#
# - When +dest+ is the path to a non-existent file,
# creates a symbolic link at +dest+ pointing to +src+:
#
# FileUtils.touch('src0.txt')
# File.exist?('dest0.txt') # => false
# FileUtils.ln_s('src0.txt', 'dest0.txt')
# File.symlink?('dest0.txt') # => true
#
# - When +dest+ is the path to an existing file,
# creates a symbolic link at +dest+ pointing to +src+
# if and only if keyword argument <tt>force: true</tt> is given
# (raises an exception otherwise):
#
# FileUtils.touch('src1.txt')
# FileUtils.touch('dest1.txt')
# FileUtils.ln_s('src1.txt', 'dest1.txt', force: true)
# FileTest.symlink?('dest1.txt') # => true
#
# FileUtils.ln_s('src1.txt', 'dest1.txt') # Raises Errno::EEXIST.
#
2022-06-15 23:56:36 +03:00
# If +dest+ is the path to a directory,
2022-05-23 16:00:21 +03:00
# creates a symbolic link at <tt>dest/src</tt> pointing to +src+:
#
# FileUtils.touch('src2.txt')
# FileUtils.mkdir('destdir2')
# FileUtils.ln_s('src2.txt', 'destdir2')
# File.symlink?('destdir2/src2.txt') # => true
#
2022-06-15 23:56:36 +03:00
# If +src+ is an array of paths to existing files and +dest+ is a directory,
2022-05-23 16:00:21 +03:00
# for each child +child+ in +src+ creates a symbolic link <tt>dest/child</tt>
# pointing to +child+:
2009-03-06 06:56:38 +03:00
#
2022-05-23 16:00:21 +03:00
# FileUtils.mkdir('srcdir3')
# FileUtils.touch('srcdir3/src0.txt')
# FileUtils.touch('srcdir3/src1.txt')
# FileUtils.mkdir('destdir3')
# FileUtils.ln_s(['srcdir3/src0.txt', 'srcdir3/src1.txt'], 'destdir3')
# File.symlink?('destdir3/src0.txt') # => true
# File.symlink?('destdir3/src1.txt') # => true
2009-03-06 06:56:38 +03:00
#
2022-05-23 16:00:21 +03:00
# Keyword arguments:
#
# - <tt>force: true</tt> - overwrites +dest+ if it exists.
2022-11-25 04:03:51 +03:00
# - <tt>relative: false</tt> - create links relative to +dest+.
2022-05-23 16:00:21 +03:00
# - <tt>noop: true</tt> - does not create links.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.ln_s('src0.txt', 'dest0.txt', noop: true, verbose: true)
# FileUtils.ln_s('src1.txt', 'destdir1', noop: true, verbose: true)
# FileUtils.ln_s('src2.txt', 'dest2.txt', force: true, noop: true, verbose: true)
# FileUtils.ln_s(['srcdir3/src0.txt', 'srcdir3/src1.txt'], 'destdir3', noop: true, verbose: true)
#
# Output:
2009-03-06 06:56:38 +03:00
#
2022-05-23 16:00:21 +03:00
# ln -s src0.txt dest0.txt
# ln -s src1.txt destdir1
# ln -sf src2.txt dest2.txt
# ln -s srcdir3/src0.txt srcdir3/src1.txt destdir3
2003-01-24 19:13:33 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.ln_sf.
#
2022-11-25 04:03:51 +03:00
def ln_s ( src , dest , force : nil , relative : false , target_directory : true , noop : nil , verbose : nil )
if relative
return ln_sr ( src , dest , force : force , noop : noop , verbose : verbose )
end
2016-03-01 05:26:44 +03:00
fu_output_message " ln -s #{ force ? 'f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
2003-11-18 08:09:20 +03:00
fu_each_src_dest0 ( src , dest ) do | s , d |
2016-03-01 05:26:44 +03:00
remove_file d , true if force
2002-03-26 05:03:04 +03:00
File . symlink s , d
end
end
2013-03-01 06:09:42 +04:00
module_function :ln_s
2002-03-26 05:03:04 +03:00
alias symlink ln_s
2013-03-01 06:09:42 +04:00
module_function :symlink
2002-03-26 05:03:04 +03:00
2022-05-23 19:17:07 +03:00
# Like FileUtils.ln_s, but always with keyword argument <tt>force: true</tt> given.
2009-03-06 06:56:38 +03:00
#
2016-03-01 05:26:44 +03:00
def ln_sf ( src , dest , noop : nil , verbose : nil )
ln_s src , dest , force : true , noop : noop , verbose : verbose
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :ln_sf
2002-03-26 05:03:04 +03:00
2022-11-25 04:03:51 +03:00
# Like FileUtils.ln_s, but create links relative to +dest+.
#
def ln_sr ( src , dest , target_directory : true , force : nil , noop : nil , verbose : nil )
options = " #{ force ? 'f' : '' } #{ target_directory ? '' : 'T' } "
dest = File . path ( dest )
srcs = Array ( src )
link = proc do | s , target_dir_p = true |
s = File . path ( s )
if target_dir_p
d = File . join ( destdirs = dest , File . basename ( s ) )
else
destdirs = File . dirname ( d = dest )
end
destdirs = fu_split_path ( File . realpath ( destdirs ) )
if fu_starting_path? ( s )
srcdirs = fu_split_path ( ( File . realdirpath ( s ) rescue File . expand_path ( s ) ) )
base = fu_relative_components_from ( srcdirs , destdirs )
s = File . join ( * base )
else
srcdirs = fu_clean_components ( * fu_split_path ( s ) )
base = fu_relative_components_from ( fu_split_path ( Dir . pwd ) , destdirs )
while srcdirs . first & . == " .. " and base . last & . != ( " .. " ) and ! fu_starting_path? ( base . last )
srcdirs . shift
base . pop
end
s = File . join ( * base , * srcdirs )
end
fu_output_message " ln -s #{ options } #{ s } #{ d } " if verbose
next if noop
remove_file d , true if force
File . symlink s , d
end
case srcs . size
when 0
when 1
link [ srcs [ 0 ] , target_directory && File . directory? ( dest ) ]
else
srcs . each ( & link )
end
end
module_function :ln_sr
2022-05-23 19:17:07 +03:00
# Creates {hard links}[https://en.wikipedia.org/wiki/Hard_link]; returns +nil+.
2018-03-13 11:18:03 +03:00
#
2022-06-13 18:39:41 +03:00
# Arguments +src+ and +dest+
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-23 19:17:07 +03:00
# If +src+ is the path to a file and +dest+ does not exist,
# creates a hard link at +dest+ pointing to +src+:
2018-03-13 11:18:03 +03:00
#
2022-05-23 19:17:07 +03:00
# FileUtils.touch('src0.txt')
2022-06-28 18:39:44 +03:00
# File.exist?('dest0.txt') # => false
2022-05-23 19:17:07 +03:00
# FileUtils.link_entry('src0.txt', 'dest0.txt')
2022-06-28 18:39:44 +03:00
# File.file?('dest0.txt') # => true
2018-03-13 11:18:03 +03:00
#
2022-05-23 19:17:07 +03:00
# If +src+ is the path to a directory and +dest+ does not exist,
# recursively creates hard links at +dest+ pointing to paths in +src+:
#
# FileUtils.mkdir_p(['src1/dir0', 'src1/dir1'])
# src_file_paths = [
# 'src1/dir0/t0.txt',
# 'src1/dir0/t1.txt',
# 'src1/dir1/t2.txt',
# 'src1/dir1/t3.txt',
# ]
# FileUtils.touch(src_file_paths)
2022-06-28 18:39:44 +03:00
# File.directory?('dest1') # => true
2022-05-23 19:17:07 +03:00
# FileUtils.link_entry('src1', 'dest1')
2022-06-28 18:39:44 +03:00
# File.file?('dest1/dir0/t0.txt') # => true
# File.file?('dest1/dir0/t1.txt') # => true
# File.file?('dest1/dir1/t2.txt') # => true
# File.file?('dest1/dir1/t3.txt') # => true
2022-05-23 19:17:07 +03:00
#
# Keyword arguments:
#
# - <tt>dereference_root: true</tt> - dereferences +src+ if it is a symbolic link.
# - <tt>remove_destination: true</tt> - removes +dest+ before creating links.
2018-03-13 11:18:03 +03:00
#
2022-05-23 19:17:07 +03:00
# Raises an exception if +dest+ is the path to an existing file or directory
# and keyword argument <tt>remove_destination: true</tt> is not given.
2018-03-13 11:18:03 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.ln (has different options).
#
2018-03-13 11:18:03 +03:00
def link_entry ( src , dest , dereference_root = false , remove_destination = false )
Entry_ . new ( src , nil , dereference_root ) . traverse do | ent |
destent = Entry_ . new ( dest , ent . rel , false )
File . unlink destent . path if remove_destination && File . file? ( destent . path )
ent . link destent . path
end
end
module_function :link_entry
2022-06-15 23:56:36 +03:00
# Copies files.
2003-01-24 19:13:33 +03:00
#
2022-06-14 16:52:18 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-24 01:57:48 +03:00
# If +src+ is the path to a file and +dest+ is not the path to a directory,
# copies +src+ to +dest+:
2003-01-24 19:13:33 +03:00
#
2022-05-24 01:57:48 +03:00
# FileUtils.touch('src0.txt')
# File.exist?('dest0.txt') # => false
# FileUtils.cp('src0.txt', 'dest0.txt')
2022-06-28 18:39:44 +03:00
# File.file?('dest0.txt') # => true
2022-05-24 01:57:48 +03:00
#
# If +src+ is the path to a file and +dest+ is the path to a directory,
# copies +src+ to <tt>dest/src</tt>:
#
# FileUtils.touch('src1.txt')
# FileUtils.mkdir('dest1')
# FileUtils.cp('src1.txt', 'dest1')
2022-06-28 18:39:44 +03:00
# File.file?('dest1/src1.txt') # => true
2022-05-24 01:57:48 +03:00
#
# If +src+ is an array of paths to files and +dest+ is the path to a directory,
# copies from each +src+ to +dest+:
#
# src_file_paths = ['src2.txt', 'src2.dat']
# FileUtils.touch(src_file_paths)
# FileUtils.mkdir('dest2')
# FileUtils.cp(src_file_paths, 'dest2')
2022-06-28 18:39:44 +03:00
# File.file?('dest2/src2.txt') # => true
# File.file?('dest2/src2.dat') # => true
2022-05-24 01:57:48 +03:00
#
# Keyword arguments:
#
# - <tt>preserve: true</tt> - preserves file times.
# - <tt>noop: true</tt> - does not copy files.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.cp('src0.txt', 'dest0.txt', noop: true, verbose: true)
# FileUtils.cp('src1.txt', 'dest1', noop: true, verbose: true)
# FileUtils.cp(src_file_paths, 'dest2', noop: true, verbose: true)
#
# Output:
#
# cp src0.txt dest0.txt
# cp src1.txt dest1
# cp src2.txt src2.dat dest2
#
# Raises an exception if +src+ is a directory.
2003-01-24 19:13:33 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2016-03-01 05:26:44 +03:00
def cp ( src , dest , preserve : nil , noop : nil , verbose : nil )
fu_output_message " cp #{ preserve ? ' -p' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
2016-03-01 05:26:44 +03:00
copy_file s , d , preserve
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
module_function :cp
2002-03-26 05:03:04 +03:00
alias copy cp
2013-03-01 06:09:42 +04:00
module_function :copy
2002-03-26 05:03:04 +03:00
2022-06-14 23:42:23 +03:00
# Recursively copies files.
2022-05-24 18:38:02 +03:00
#
2022-06-14 23:42:23 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
2022-05-24 18:38:02 +03:00
#
2022-06-21 20:23:06 +03:00
# The mode, owner, and group are retained in the copy;
# to change those, use FileUtils.install instead.
#
2022-05-24 18:38:02 +03:00
# If +src+ is the path to a file and +dest+ is not the path to a directory,
# copies +src+ to +dest+:
#
# FileUtils.touch('src0.txt')
# File.exist?('dest0.txt') # => false
# FileUtils.cp_r('src0.txt', 'dest0.txt')
2022-06-28 18:39:44 +03:00
# File.file?('dest0.txt') # => true
2022-05-24 18:38:02 +03:00
#
# If +src+ is the path to a file and +dest+ is the path to a directory,
# copies +src+ to <tt>dest/src</tt>:
2009-03-06 06:56:38 +03:00
#
2022-05-24 18:38:02 +03:00
# FileUtils.touch('src1.txt')
# FileUtils.mkdir('dest1')
# FileUtils.cp_r('src1.txt', 'dest1')
2022-06-28 18:39:44 +03:00
# File.file?('dest1/src1.txt') # => true
2022-05-24 18:38:02 +03:00
#
# If +src+ is the path to a directory and +dest+ does not exist,
# recursively copies +src+ to +dest+:
#
2022-06-14 23:42:23 +03:00
# tree('src2')
2022-06-28 18:39:44 +03:00
# # => src2
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-06-14 23:42:23 +03:00
# FileUtils.exist?('dest2') # => false
2022-05-24 18:38:02 +03:00
# FileUtils.cp_r('src2', 'dest2')
2022-06-14 23:42:23 +03:00
# tree('dest2')
2022-06-28 18:39:44 +03:00
# # => dest2
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-05-24 18:38:02 +03:00
#
# If +src+ and +dest+ are paths to directories,
# recursively copies +src+ to <tt>dest/src</tt>:
#
2022-06-14 23:42:23 +03:00
# tree('src3')
2022-06-28 18:39:44 +03:00
# # => src3
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-05-24 18:38:02 +03:00
# FileUtils.mkdir('dest3')
# FileUtils.cp_r('src3', 'dest3')
2022-06-14 23:42:23 +03:00
# tree('dest3')
2022-06-28 18:39:44 +03:00
# # => dest3
# # `-- src3
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-06-14 23:42:23 +03:00
#
# If +src+ is an array of paths and +dest+ is a directory,
# recursively copies from each path in +src+ to +dest+;
# the paths in +src+ may point to files and/or directories.
2022-05-24 18:38:02 +03:00
#
# Keyword arguments:
2003-01-24 19:13:33 +03:00
#
2022-05-24 18:38:02 +03:00
# - <tt>dereference_root: false</tt> - if +src+ is a symbolic link,
# does not dereference it.
# - <tt>noop: true</tt> - does not copy files.
2022-05-30 20:29:54 +03:00
# - <tt>preserve: true</tt> - preserves file times.
2022-05-24 18:38:02 +03:00
# - <tt>remove_destination: true</tt> - removes +dest+ before copying files.
# - <tt>verbose: true</tt> - prints an equivalent command:
2009-03-06 06:56:38 +03:00
#
2022-05-24 18:38:02 +03:00
# FileUtils.cp_r('src0.txt', 'dest0.txt', noop: true, verbose: true)
# FileUtils.cp_r('src1.txt', 'dest1', noop: true, verbose: true)
# FileUtils.cp_r('src2', 'dest2', noop: true, verbose: true)
# FileUtils.cp_r('src3', 'dest3', noop: true, verbose: true)
2019-03-10 14:09:59 +03:00
#
2022-05-24 18:38:02 +03:00
# Output:
2019-03-10 14:09:59 +03:00
#
2022-05-24 18:38:02 +03:00
# cp -r src0.txt dest0.txt
# cp -r src1.txt dest1
# cp -r src2 dest2
# cp -r src3 dest3
2009-03-06 06:56:38 +03:00
#
2022-05-24 18:38:02 +03:00
# Raises an exception of +src+ is the path to a directory
# and +dest+ is the path to a file.
2004-08-09 01:35:11 +04:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
2009-03-06 06:56:38 +03:00
#
2016-03-01 05:26:44 +03:00
def cp_r ( src , dest , preserve : nil , noop : nil , verbose : nil ,
dereference_root : true , remove_destination : nil )
fu_output_message " cp -r #{ preserve ? 'p' : '' } #{ remove_destination ? ' --remove-destination' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
2016-03-01 05:26:44 +03:00
copy_entry s , d , preserve , dereference_root , remove_destination
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
module_function :cp_r
2002-03-26 05:03:04 +03:00
2022-05-26 01:08:14 +03:00
# Recursively copies files from +src+ to +dest+.
2004-08-09 01:35:11 +04:00
#
2022-06-13 18:39:41 +03:00
# Arguments +src+ and +dest+
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-05-26 01:08:14 +03:00
# If +src+ is the path to a file, copies +src+ to +dest+:
2004-08-09 01:35:11 +04:00
#
2022-05-26 01:08:14 +03:00
# FileUtils.touch('src0.txt')
# File.exist?('dest0.txt') # => false
# FileUtils.copy_entry('src0.txt', 'dest0.txt')
# File.file?('dest0.txt') # => true
#
# If +src+ is a directory, recursively copies +src+ to +dest+:
#
2022-06-08 15:34:47 +03:00
# tree('src1')
2022-06-28 18:39:44 +03:00
# # => src1
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-05-26 01:08:14 +03:00
# FileUtils.copy_entry('src1', 'dest1')
2022-06-08 15:34:47 +03:00
# tree('dest1')
2022-06-28 18:39:44 +03:00
# # => dest1
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-05-26 01:08:14 +03:00
#
# The recursive copying preserves file types for regular files,
# directories, and symbolic links;
# other file types (FIFO streams, device files, etc.) are not supported.
2004-08-09 01:35:11 +04:00
#
2022-05-26 01:08:14 +03:00
# Keyword arguments:
2005-05-26 02:41:32 +04:00
#
2022-05-26 01:08:14 +03:00
# - <tt>dereference_root: true</tt> - if +src+ is a symbolic link,
# follows the link.
2022-05-30 20:29:54 +03:00
# - <tt>preserve: true</tt> - preserves file times.
2022-05-26 01:08:14 +03:00
# - <tt>remove_destination: true</tt> - removes +dest+ before copying files.
2006-03-03 15:24:53 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2006-03-03 15:24:53 +03:00
def copy_entry ( src , dest , preserve = false , dereference_root = false , remove_destination = false )
2019-08-24 00:52:12 +03:00
if dereference_root
src = File . realpath ( src )
end
Entry_ . new ( src , nil , false ) . wrap_traverse ( proc do | ent |
2005-05-26 02:41:32 +04:00
destent = Entry_ . new ( dest , ent . rel , false )
2017-10-20 10:03:11 +03:00
File . unlink destent . path if remove_destination && ( File . file? ( destent . path ) || File . symlink? ( destent . path ) )
2005-05-26 02:41:32 +04:00
ent . copy destent . path
2013-02-02 07:54:00 +04:00
end , proc do | ent |
destent = Entry_ . new ( dest , ent . rel , false )
2005-05-26 02:41:32 +04:00
ent . copy_metadata destent . path if preserve
2013-02-02 07:54:00 +04:00
end )
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :copy_entry
2002-03-26 05:03:04 +03:00
2022-06-13 18:39:41 +03:00
# Copies file from +src+ to +dest+, which should not be directories.
#
# Arguments +src+ and +dest+
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
# Examples:
2022-05-30 20:29:54 +03:00
#
# FileUtils.touch('src0.txt')
# FileUtils.copy_file('src0.txt', 'dest0.txt')
# File.file?('dest0.txt') # => true
2003-02-21 14:57:31 +03:00
#
2022-05-30 20:29:54 +03:00
# Keyword arguments:
#
# - <tt>dereference: false</tt> - if +src+ is a symbolic link,
# does not follow the link.
# - <tt>preserve: true</tt> - preserves file times.
# - <tt>remove_destination: true</tt> - removes +dest+ before copying files.
2003-02-21 14:57:31 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2004-08-09 01:35:11 +04:00
def copy_file ( src , dest , preserve = false , dereference = true )
2005-05-26 02:41:32 +04:00
ent = Entry_ . new ( src , nil , dereference )
ent . copy_file dest
ent . copy_metadata dest if preserve
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :copy_file
2002-03-26 05:03:04 +03:00
2022-05-30 20:29:54 +03:00
# Copies \IO stream +src+ to \IO stream +dest+ via
2022-10-12 05:54:16 +03:00
# {IO.copy_stream}[rdoc-ref:IO.copy_stream].
2003-02-21 14:57:31 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2003-12-11 12:49:47 +03:00
def copy_stream ( src , dest )
2008-09-01 06:31:56 +04:00
IO . copy_stream ( src , dest )
2004-08-09 01:35:11 +04:00
end
2013-03-01 06:09:42 +04:00
module_function :copy_stream
2004-08-09 01:35:11 +04:00
2022-06-15 23:56:36 +03:00
# Moves entries.
2022-05-30 20:29:54 +03:00
#
2022-06-15 23:56:36 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-15 23:56:36 +03:00
# If +src+ and +dest+ are on different file systems,
# first copies, then removes +src+.
#
2022-06-06 18:37:18 +03:00
# May cause a local vulnerability if not called with keyword argument
# <tt>secure: true</tt>;
# see {Avoiding the TOCTTOU Vulnerability}[rdoc-ref:FileUtils@Avoiding+the+TOCTTOU+Vulnerability].
#
2022-05-30 20:29:54 +03:00
# If +src+ is the path to a single file or directory and +dest+ does not exist,
# moves +src+ to +dest+:
#
2022-06-08 15:34:47 +03:00
# tree('src0')
2022-06-28 18:39:44 +03:00
# # => src0
# # |-- src0.txt
# # `-- src1.txt
2022-05-30 20:29:54 +03:00
# File.exist?('dest0') # => false
# FileUtils.mv('src0', 'dest0')
# File.exist?('src0') # => false
2022-06-08 15:34:47 +03:00
# tree('dest0')
2022-06-28 18:39:44 +03:00
# # => dest0
# # |-- src0.txt
# # `-- src1.txt
2022-05-30 20:29:54 +03:00
#
# If +src+ is an array of paths to files and directories
# and +dest+ is the path to a directory,
# copies from each path in the array to +dest+:
#
# File.file?('src1.txt') # => true
2022-06-08 15:34:47 +03:00
# tree('src1')
2022-06-28 18:39:44 +03:00
# # => src1
# # |-- src.dat
# # `-- src.txt
# Dir.empty?('dest1') # => true
2022-05-30 20:29:54 +03:00
# FileUtils.mv(['src1.txt', 'src1'], 'dest1')
2022-06-08 15:34:47 +03:00
# tree('dest1')
2022-06-28 18:39:44 +03:00
# # => dest1
# # |-- src1
# # | |-- src.dat
# # | `-- src.txt
# # `-- src1.txt
2022-05-30 20:29:54 +03:00
#
2022-06-06 18:37:18 +03:00
# Keyword arguments:
#
# - <tt>force: true</tt> - if the move includes removing +src+
2022-06-15 23:56:36 +03:00
# (that is, if +src+ and +dest+ are on different file systems),
2022-05-30 20:29:54 +03:00
# ignores raised exceptions of StandardError and its descendants.
# - <tt>noop: true</tt> - does not move files.
2022-06-06 18:37:18 +03:00
# - <tt>secure: true</tt> - removes +src+ securely;
# see details at FileUtils.remove_entry_secure.
2022-05-30 20:29:54 +03:00
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.mv('src0', 'dest0', noop: true, verbose: true)
# FileUtils.mv(['src1.txt', 'src1'], 'dest1', noop: true, verbose: true)
2009-03-06 06:56:38 +03:00
#
2022-05-30 20:29:54 +03:00
# Output:
2009-03-06 06:56:38 +03:00
#
2022-05-30 20:29:54 +03:00
# mv src0 dest0
# mv src1.txt src1 dest1
2009-03-06 06:56:38 +03:00
#
2016-03-01 05:26:44 +03:00
def mv ( src , dest , force : nil , noop : nil , verbose : nil , secure : nil )
fu_output_message " mv #{ force ? ' -f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if verbose
return if noop
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
destent = Entry_ . new ( d , nil , true )
2004-12-27 09:15:32 +03:00
begin
2005-05-26 02:41:32 +04:00
if destent . exist?
if destent . directory?
2015-04-01 03:59:14 +03:00
raise Errno :: EEXIST , d
2005-05-26 02:41:32 +04:00
end
2004-12-27 09:15:32 +03:00
end
2004-05-07 17:08:12 +04:00
begin
2004-12-27 09:15:32 +03:00
File . rename s , d
2019-03-11 12:52:31 +03:00
rescue Errno :: EXDEV ,
Errno :: EPERM # move from unencrypted to encrypted dir (ext4)
2004-08-09 01:35:11 +04:00
copy_entry s , d , true
2016-03-01 05:26:44 +03:00
if secure
remove_entry_secure s , force
2007-03-03 16:53:38 +03:00
else
2016-03-01 05:26:44 +03:00
remove_entry s , force
2007-03-03 16:53:38 +03:00
end
2002-03-26 05:03:04 +03:00
end
2004-12-27 09:15:32 +03:00
rescue SystemCallError
2016-03-01 05:26:44 +03:00
raise unless force
2002-03-26 05:03:04 +03:00
end
end
end
2013-03-01 06:09:42 +04:00
module_function :mv
2002-03-26 05:03:04 +03:00
alias move mv
2013-03-01 06:09:42 +04:00
module_function :move
2002-03-26 05:03:04 +03:00
2022-06-14 16:52:18 +03:00
# Removes entries at the paths in the given +list+
# (a single path or an array of paths)
2022-06-06 18:37:18 +03:00
# returns +list+, if it is an array, <tt>[list]</tt> otherwise.
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-06 18:37:18 +03:00
# With no keyword arguments, removes files at the paths given in +list+:
#
# FileUtils.touch(['src0.txt', 'src0.dat'])
# FileUtils.rm(['src0.dat', 'src0.txt']) # => ["src0.dat", "src0.txt"]
#
# Keyword arguments:
#
2022-08-30 10:02:02 +03:00
# - <tt>force: true</tt> - ignores raised exceptions of StandardError
2022-06-06 18:37:18 +03:00
# and its descendants.
# - <tt>noop: true</tt> - does not remove files; returns +nil+.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.rm(['src0.dat', 'src0.txt'], noop: true, verbose: true)
#
# Output:
2009-03-06 06:56:38 +03:00
#
2022-06-06 18:37:18 +03:00
# rm src0.dat src0.txt
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2016-03-01 05:26:44 +03:00
def rm ( list , force : nil , noop : nil , verbose : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message " rm #{ force ? ' -f' : '' } #{ list . join ' ' } " if verbose
return if noop
2002-03-26 05:03:04 +03:00
2005-05-21 21:40:24 +04:00
list . each do | path |
2016-03-01 05:26:44 +03:00
remove_file path , force
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
module_function :rm
2002-03-26 05:03:04 +03:00
alias remove rm
2013-03-01 06:09:42 +04:00
module_function :remove
2002-03-26 05:03:04 +03:00
2022-06-06 18:37:18 +03:00
# Equivalent to:
2009-03-06 06:56:38 +03:00
#
2022-06-06 18:37:18 +03:00
# FileUtils.rm(list, force: true, **kwargs)
2005-05-26 02:41:32 +04:00
#
2022-06-16 17:56:33 +03:00
# Argument +list+ (a single path or an array of paths)
2022-06-13 18:39:41 +03:00
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-06 18:37:18 +03:00
# See FileUtils.rm for keyword arguments.
#
2022-06-21 16:42:27 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2016-03-01 05:26:44 +03:00
def rm_f ( list , noop : nil , verbose : nil )
rm list , force : true , noop : noop , verbose : verbose
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :rm_f
2002-03-26 05:03:04 +03:00
alias safe_unlink rm_f
2013-03-01 06:09:42 +04:00
module_function :safe_unlink
2002-03-26 05:03:04 +03:00
2022-06-06 18:37:18 +03:00
# Removes entries at the paths in the given +list+
2022-06-14 16:52:18 +03:00
# (a single path or an array of paths);
2022-06-06 18:37:18 +03:00
# returns +list+, if it is an array, <tt>[list]</tt> otherwise.
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-06 18:37:18 +03:00
# May cause a local vulnerability if not called with keyword argument
# <tt>secure: true</tt>;
# see {Avoiding the TOCTTOU Vulnerability}[rdoc-ref:FileUtils@Avoiding+the+TOCTTOU+Vulnerability].
#
# For each file path, removes the file at that path:
2009-03-06 06:56:38 +03:00
#
2022-06-06 18:37:18 +03:00
# FileUtils.touch(['src0.txt', 'src0.dat'])
# FileUtils.rm_r(['src0.dat', 'src0.txt'])
# File.exist?('src0.txt') # => false
# File.exist?('src0.dat') # => false
2009-03-06 06:56:38 +03:00
#
2022-06-06 18:37:18 +03:00
# For each directory path, recursively removes files and directories:
2005-05-21 21:40:24 +04:00
#
2022-06-08 15:34:47 +03:00
# tree('src1')
2022-06-28 18:39:44 +03:00
# # => src1
# # |-- dir0
# # | |-- src0.txt
# # | `-- src1.txt
# # `-- dir1
# # |-- src2.txt
# # `-- src3.txt
2022-06-06 18:37:18 +03:00
# FileUtils.rm_r('src1')
# File.exist?('src1') # => false
2005-05-25 14:50:14 +04:00
#
2022-06-06 18:37:18 +03:00
# Keyword arguments:
#
2022-08-30 10:02:02 +03:00
# - <tt>force: true</tt> - ignores raised exceptions of StandardError
2022-06-06 18:37:18 +03:00
# and its descendants.
# - <tt>noop: true</tt> - does not remove entries; returns +nil+.
# - <tt>secure: true</tt> - removes +src+ securely;
# see details at FileUtils.remove_entry_secure.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.rm_r(['src0.dat', 'src0.txt'], noop: true, verbose: true)
# FileUtils.rm_r('src1', noop: true, verbose: true)
#
# Output:
#
# rm -r src0.dat src0.txt
# rm -r src1
2009-03-06 06:56:38 +03:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2016-03-01 05:26:44 +03:00
def rm_r ( list , force : nil , noop : nil , verbose : nil , secure : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message " rm -r #{ force ? 'f' : '' } #{ list . join ' ' } " if verbose
return if noop
2005-05-21 21:40:24 +04:00
list . each do | path |
2016-03-01 05:26:44 +03:00
if secure
remove_entry_secure path , force
2005-05-21 21:40:24 +04:00
else
2016-03-01 05:26:44 +03:00
remove_entry path , force
2005-05-21 21:40:24 +04:00
end
end
end
2013-03-01 06:09:42 +04:00
module_function :rm_r
2005-05-21 21:40:24 +04:00
2022-06-06 18:37:18 +03:00
# Equivalent to:
#
# FileUtils.rm_r(list, force: true, **kwargs)
2009-03-06 06:56:38 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-06 18:37:18 +03:00
# May cause a local vulnerability if not called with keyword argument
# <tt>secure: true</tt>;
# see {Avoiding the TOCTTOU Vulnerability}[rdoc-ref:FileUtils@Avoiding+the+TOCTTOU+Vulnerability].
2005-05-26 02:41:32 +04:00
#
2022-06-06 18:37:18 +03:00
# See FileUtils.rm_r for keyword arguments.
2005-05-25 14:50:14 +04:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2016-03-01 05:26:44 +03:00
def rm_rf ( list , noop : nil , verbose : nil , secure : nil )
rm_r list , force : true , noop : noop , verbose : verbose , secure : secure
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :rm_rf
2002-03-26 05:03:04 +03:00
2003-02-21 14:57:31 +03:00
alias rmtree rm_rf
2013-03-01 06:09:42 +04:00
module_function :rmtree
2003-02-21 14:57:31 +03:00
2022-06-06 18:37:18 +03:00
# Securely removes the entry given by +path+,
# which should be the entry for a regular file, a symbolic link,
# or a directory.
2005-05-26 02:41:32 +04:00
#
2022-06-13 18:39:41 +03:00
# Argument +path+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-06 18:37:18 +03:00
# Avoids a local vulnerability that can exist in certain circumstances;
# see {Avoiding the TOCTTOU Vulnerability}[rdoc-ref:FileUtils@Avoiding+the+TOCTTOU+Vulnerability].
2005-05-26 02:41:32 +04:00
#
2022-06-08 23:37:57 +03:00
# Optional argument +force+ specifies whether to ignore
2022-08-30 10:02:02 +03:00
# raised exceptions of StandardError and its descendants.
2022-06-08 23:37:57 +03:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2005-05-26 02:41:32 +04:00
def remove_entry_secure ( path , force = false )
2005-09-19 00:33:01 +04:00
unless fu_have_symlink?
remove_entry path , force
return
end
2005-08-13 17:06:04 +04:00
fullpath = File . expand_path ( path )
2005-09-19 00:33:01 +04:00
st = File . lstat ( fullpath )
unless st . directory?
File . unlink fullpath
return
end
# is a directory.
parent_st = File . stat ( File . dirname ( fullpath ) )
unless parent_st . world_writable?
2005-08-13 17:06:04 +04:00
remove_entry path , force
2005-05-26 15:33:16 +04:00
return
end
2005-09-19 00:33:01 +04:00
unless parent_st . sticky?
raise ArgumentError , " parent directory is world writable, FileUtils # remove_entry_secure does not work; abort: #{ path . inspect } (parent directory mode #{ '%o' % parent_st . mode } ) "
2005-08-13 17:06:04 +04:00
end
2018-05-15 08:53:18 +03:00
2005-09-19 00:33:01 +04:00
# freeze tree root
2005-08-13 17:06:04 +04:00
euid = Process . euid
2018-05-15 08:53:18 +03:00
dot_file = fullpath + " /. "
begin
File . open ( dot_file ) { | f |
unless fu_stat_identical_entry? ( st , f . stat )
# symlink (TOC-to-TOU attack?)
File . unlink fullpath
return
end
f . chown euid , - 1
f . chmod 0700
}
2019-01-31 17:02:30 +03:00
rescue Errno :: EISDIR # JRuby in non-native mode can't open files as dirs
2018-05-15 08:53:18 +03:00
File . lstat ( dot_file ) . tap { | fstat |
unless fu_stat_identical_entry? ( st , fstat )
# symlink (TOC-to-TOU attack?)
File . unlink fullpath
return
end
File . chown euid , - 1 , dot_file
File . chmod 0700 , dot_file
}
end
unless fu_stat_identical_entry? ( st , File . lstat ( fullpath ) )
# TOC-to-TOU attack?
File . unlink fullpath
return
end
2005-08-13 17:06:04 +04:00
# ---- tree root is frozen ----
2005-05-26 02:41:32 +04:00
root = Entry_ . new ( path )
root . preorder_traverse do | ent |
if ent . directory?
2005-09-19 00:33:01 +04:00
ent . chown euid , - 1
2005-05-26 02:41:32 +04:00
ent . chmod 0700
2002-03-26 05:03:04 +03:00
end
end
2005-05-26 02:41:32 +04:00
root . postorder_traverse do | ent |
begin
ent . remove
rescue
raise unless force
end
end
rescue
raise unless force
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :remove_entry_secure
2002-03-26 05:03:04 +03:00
2011-05-13 02:49:40 +04:00
def fu_have_symlink? #:nodoc:
2005-09-19 00:33:01 +04:00
File . symlink nil , nil
rescue NotImplementedError
return false
2012-03-12 15:09:11 +04:00
rescue TypeError
2005-09-19 00:33:01 +04:00
return true
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_have_symlink?
2005-09-19 00:33:01 +04:00
2005-09-19 00:59:29 +04:00
def fu_stat_identical_entry? ( a , b ) #:nodoc:
2005-08-13 17:06:04 +04:00
a . dev == b . dev and a . ino == b . ino
2005-05-26 02:41:32 +04:00
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_stat_identical_entry?
2005-05-26 02:41:32 +04:00
2022-06-08 23:37:57 +03:00
# Removes the entry given by +path+,
# which should be the entry for a regular file, a symbolic link,
# or a directory.
2005-05-26 02:41:32 +04:00
#
2022-06-13 18:39:41 +03:00
# Argument +path+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-08 23:37:57 +03:00
# Optional argument +force+ specifies whether to ignore
2022-08-30 10:02:02 +03:00
# raised exceptions of StandardError and its descendants.
2005-05-26 02:41:32 +04:00
#
2022-06-08 23:37:57 +03:00
# Related: FileUtils.remove_entry_secure.
2005-05-26 02:41:32 +04:00
#
def remove_entry ( path , force = false )
Entry_ . new ( path ) . postorder_traverse do | ent |
begin
ent . remove
rescue
raise unless force
2005-04-14 14:00:52 +04:00
end
2002-03-26 05:03:04 +03:00
end
2005-05-26 02:41:32 +04:00
rescue
raise unless force
end
2013-03-01 06:09:42 +04:00
module_function :remove_entry
2005-05-26 02:41:32 +04:00
2022-06-08 23:37:57 +03:00
# Removes the file entry given by +path+,
# which should be the entry for a regular file or a symbolic link.
2005-05-26 02:41:32 +04:00
#
2022-06-13 18:39:41 +03:00
# Argument +path+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-08 23:37:57 +03:00
# Optional argument +force+ specifies whether to ignore
2022-08-30 10:02:02 +03:00
# raised exceptions of StandardError and its descendants.
2005-05-26 02:41:32 +04:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2005-05-26 02:41:32 +04:00
def remove_file ( path , force = false )
Entry_ . new ( path ) . remove_file
rescue
raise unless force
end
2013-03-01 06:09:42 +04:00
module_function :remove_file
2005-05-26 02:41:32 +04:00
2022-06-08 23:37:57 +03:00
# Recursively removes the directory entry given by +path+,
# which should be the entry for a regular file, a symbolic link,
# or a directory.
2005-05-26 02:41:32 +04:00
#
2022-06-13 18:39:41 +03:00
# Argument +path+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-08 23:37:57 +03:00
# Optional argument +force+ specifies whether to ignore
2022-08-30 10:02:02 +03:00
# raised exceptions of StandardError and its descendants.
2005-05-26 02:41:32 +04:00
#
2022-07-29 09:47:02 +03:00
# Related: {methods for deleting}[rdoc-ref:FileUtils@Deleting].
#
2005-05-26 02:41:32 +04:00
def remove_dir ( path , force = false )
remove_entry path , force # FIXME?? check if it is a directory
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :remove_dir
2002-03-26 05:03:04 +03:00
2022-06-08 23:37:57 +03:00
# Returns +true+ if the contents of files +a+ and +b+ are identical,
# +false+ otherwise.
2003-01-24 19:13:33 +03:00
#
2022-06-13 18:39:41 +03:00
# Arguments +a+ and +b+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-16 17:56:33 +03:00
# FileUtils.identical? and FileUtils.cmp are aliases for FileUtils.compare_file.
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.compare_stream.
#
2003-12-11 12:49:47 +03:00
def compare_file ( a , b )
2003-02-21 14:57:31 +03:00
return false unless File . size ( a ) == File . size ( b )
File . open ( a , 'rb' ) { | fa |
2003-12-12 13:33:24 +03:00
File . open ( b , 'rb' ) { | fb |
2003-02-21 14:57:31 +03:00
return compare_stream ( fa , fb )
2003-12-12 13:33:24 +03:00
}
}
2003-02-21 14:57:31 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :compare_file
2002-03-26 05:03:04 +03:00
2003-02-21 14:57:31 +03:00
alias identical? compare_file
alias cmp compare_file
2013-03-01 06:09:42 +04:00
module_function :identical?
module_function :cmp
2002-03-26 05:03:04 +03:00
2022-06-08 23:37:57 +03:00
# Returns +true+ if the contents of streams +a+ and +b+ are identical,
# +false+ otherwise.
2003-02-21 14:57:31 +03:00
#
2022-06-13 18:39:41 +03:00
# Arguments +a+ and +b+
# should be {interpretable as a path}[rdoc-ref:FileUtils@Path+Arguments].
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.compare_file.
#
2003-12-11 12:49:47 +03:00
def compare_stream ( a , b )
2003-02-21 14:57:31 +03:00
bsize = fu_stream_blksize ( a , b )
2018-05-15 08:53:18 +03:00
2021-01-17 05:52:03 +03:00
sa = String . new ( capacity : bsize )
sb = String . new ( capacity : bsize )
2018-05-15 08:53:18 +03:00
2012-11-04 06:56:11 +04:00
begin
a . read ( bsize , sa )
b . read ( bsize , sb )
return true if sa . empty? && sb . empty?
end while sa == sb
2002-03-26 05:03:04 +03:00
false
end
2013-03-01 06:09:42 +04:00
module_function :compare_stream
2002-03-26 05:03:04 +03:00
2022-06-16 17:56:33 +03:00
# Copies a file entry.
# See {install(1)}[https://man7.org/linux/man-pages/man1/install.1.html].
2022-06-08 23:37:57 +03:00
#
2022-06-15 23:56:36 +03:00
# Arguments +src+ (a single path or an array of paths)
# and +dest+ (a single path)
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments];
2022-06-08 23:37:57 +03:00
#
# If the entry at +dest+ does not exist, copies from +src+ to +dest+:
#
# File.read('src0.txt') # => "aaa\n"
# File.exist?('dest0.txt') # => false
# FileUtils.install('src0.txt', 'dest0.txt')
# File.read('dest0.txt') # => "aaa\n"
#
# If +dest+ is a file entry, copies from +src+ to +dest+, overwriting:
2009-03-06 06:56:38 +03:00
#
2022-06-08 23:37:57 +03:00
# File.read('src1.txt') # => "aaa\n"
# File.read('dest1.txt') # => "bbb\n"
# FileUtils.install('src1.txt', 'dest1.txt')
# File.read('dest1.txt') # => "aaa\n"
#
# If +dest+ is a directory entry, copies from +src+ to <tt>dest/src</tt>,
# overwriting if necessary:
#
# File.read('src2.txt') # => "aaa\n"
# File.read('dest2/src2.txt') # => "bbb\n"
# FileUtils.install('src2.txt', 'dest2')
# File.read('dest2/src2.txt') # => "aaa\n"
#
2022-06-15 23:56:36 +03:00
# If +src+ is an array of paths and +dest+ points to a directory,
# copies each path +path+ in +src+ to <tt>dest/path</tt>:
#
# File.file?('src3.txt') # => true
# File.file?('src3.dat') # => true
# FileUtils.mkdir('dest3')
# FileUtils.install(['src3.txt', 'src3.dat'], 'dest3')
# File.file?('dest3/src3.txt') # => true
# File.file?('dest3/src3.dat') # => true
#
2022-06-08 23:37:57 +03:00
# Keyword arguments:
#
# - <tt>group: <i>group</i></tt> - changes the group if not +nil+,
2022-10-12 05:54:16 +03:00
# using {File.chown}[rdoc-ref:File.chown].
2022-06-08 23:37:57 +03:00
# - <tt>mode: <i>permissions</i></tt> - changes the permissions.
2022-10-12 05:54:16 +03:00
# using {File.chmod}[rdoc-ref:File.chmod].
2022-07-29 09:47:02 +03:00
# - <tt>noop: true</tt> - does not copy entries; returns +nil+.
2022-06-08 23:37:57 +03:00
# - <tt>owner: <i>owner</i></tt> - changes the owner if not +nil+,
2022-10-12 05:54:16 +03:00
# using {File.chown}[rdoc-ref:File.chown].
2022-06-08 23:37:57 +03:00
# - <tt>preserve: true</tt> - preserve timestamps
2022-10-12 05:54:16 +03:00
# using {File.utime}[rdoc-ref:File.utime].
2022-06-08 23:37:57 +03:00
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.install('src0.txt', 'dest0.txt', noop: true, verbose: true)
# FileUtils.install('src1.txt', 'dest1.txt', noop: true, verbose: true)
# FileUtils.install('src2.txt', 'dest2', noop: true, verbose: true)
#
# Output:
2009-03-06 06:56:38 +03:00
#
2022-06-08 23:37:57 +03:00
# install -c src0.txt dest0.txt
# install -c src1.txt dest1.txt
# install -c src2.txt dest2
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: {methods for copying}[rdoc-ref:FileUtils@Copying].
#
2016-06-27 10:55:17 +03:00
def install ( src , dest , mode : nil , owner : nil , group : nil , preserve : nil ,
noop : nil , verbose : nil )
if verbose
msg = + " install -c "
msg << ' -p' if preserve
2016-06-27 10:58:34 +03:00
msg << ' -m ' << mode_to_s ( mode ) if mode
2016-06-27 10:55:17 +03:00
msg << " -o #{ owner } " if owner
msg << " -g #{ group } " if group
msg << ' ' << [ src , dest ] . flatten . join ( ' ' )
fu_output_message msg
end
2016-03-01 05:26:44 +03:00
return if noop
2016-06-27 10:55:17 +03:00
uid = fu_get_uid ( owner )
gid = fu_get_gid ( group )
2014-07-11 11:51:19 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
st = File . stat ( s )
2005-05-26 02:41:32 +04:00
unless File . exist? ( d ) and compare_file ( s , d )
2003-12-12 13:33:24 +03:00
remove_file d , true
2023-02-10 06:37:26 +03:00
if d . end_with? ( '/' )
mkdir_p d
copy_file s , d + File . basename ( s )
else
mkdir_p File . expand_path ( '..' , d )
copy_file s , d
end
2016-03-01 05:26:44 +03:00
File . utime st . atime , st . mtime , d if preserve
2016-06-27 10:58:34 +03:00
File . chmod fu_mode ( mode , st ) , d if mode
2016-06-27 10:55:17 +03:00
File . chown uid , gid , d if uid or gid
2002-03-26 05:03:04 +03:00
end
end
end
2013-03-01 06:09:42 +04:00
module_function :install
2002-03-26 05:03:04 +03:00
2011-05-13 02:49:40 +04:00
def user_mask ( target ) #:nodoc:
2013-07-04 21:22:08 +04:00
target . each_char . inject ( 0 ) do | mask , chr |
case chr
when " u "
mask | 04700
when " g "
mask | 02070
when " o "
mask | 01007
when " a "
mask | 07777
else
2024-01-19 10:03:38 +03:00
raise ArgumentError , " invalid 'who' symbol in file mode: #{ chr } "
2011-05-01 20:16:05 +04:00
end
end
end
2013-03-01 06:09:42 +04:00
private_module_function :user_mask
2011-05-01 20:16:05 +04:00
2017-03-01 13:17:42 +03:00
def apply_mask ( mode , user_mask , op , mode_mask ) #:nodoc:
2013-07-04 21:22:08 +04:00
case op
when '='
( mode & ~ user_mask ) | ( user_mask & mode_mask )
when '+'
mode | ( user_mask & mode_mask )
when '-'
mode & ~ ( user_mask & mode_mask )
2011-05-01 20:16:05 +04:00
end
end
2013-07-04 21:22:08 +04:00
private_module_function :apply_mask
2011-05-01 20:16:05 +04:00
2013-07-04 21:22:08 +04:00
def symbolic_modes_to_i ( mode_sym , path ) #:nodoc:
2019-10-02 19:39:02 +03:00
path = File . stat ( path ) unless File :: Stat === path
mode = path . mode
2016-06-27 10:58:34 +03:00
mode_sym . split ( / , / ) . inject ( mode & 07777 ) do | current_mode , clause |
2013-07-04 21:22:08 +04:00
target , * actions = clause . split ( / ([=+-]) / )
raise ArgumentError , " invalid file mode: #{ mode_sym } " if actions . empty?
target = 'a' if target . empty?
2011-05-01 20:16:05 +04:00
user_mask = user_mask ( target )
2013-07-04 21:22:08 +04:00
actions . each_slice ( 2 ) do | op , perm |
need_apply = op == '='
mode_mask = ( perm || '' ) . each_char . inject ( 0 ) do | mask , chr |
case chr
when " r "
mask | 0444
when " w "
mask | 0222
when " x "
mask | 0111
when " X "
2019-10-02 19:39:02 +03:00
if path . directory?
2013-07-04 21:22:08 +04:00
mask | 0111
else
mask
end
when " s "
mask | 06000
when " t "
mask | 01000
when " u " , " g " , " o "
if mask . nonzero?
current_mode = apply_mask ( current_mode , user_mask , op , mask )
end
need_apply = false
copy_mask = user_mask ( chr )
( current_mode & copy_mask ) / ( copy_mask & 0111 ) * ( user_mask & 0111 )
else
2024-01-19 10:03:38 +03:00
raise ArgumentError , " invalid 'perm' symbol in file mode: #{ chr } "
2013-07-04 21:22:08 +04:00
end
end
if mode_mask . nonzero? || need_apply
current_mode = apply_mask ( current_mode , user_mask , op , mode_mask )
end
2011-05-01 20:16:05 +04:00
end
2013-07-04 21:22:08 +04:00
current_mode
2011-05-01 20:16:05 +04:00
end
end
2013-03-01 06:09:42 +04:00
private_module_function :symbolic_modes_to_i
2011-05-01 20:16:05 +04:00
2011-05-13 02:49:40 +04:00
def fu_mode ( mode , path ) #:nodoc:
2011-05-01 20:16:05 +04:00
mode . is_a? ( String ) ? symbolic_modes_to_i ( mode , path ) : mode
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_mode
2012-02-17 21:59:47 +04:00
2013-02-02 07:07:17 +04:00
def mode_to_s ( mode ) #:nodoc:
mode . is_a? ( String ) ? mode : " %o " % mode
end
2013-07-09 11:02:04 +04:00
private_module_function :mode_to_s
2013-02-02 07:07:17 +04:00
2022-06-09 23:03:21 +03:00
# Changes permissions on the entries at the paths given in +list+
2022-06-14 16:52:18 +03:00
# (a single path or an array of paths)
2022-06-16 17:56:33 +03:00
# to the permissions given by +mode+;
# returns +list+ if it is an array, <tt>[list]</tt> otherwise:
2022-06-09 23:03:21 +03:00
#
# - Modifies each entry that is a regular file using
2022-10-12 05:54:16 +03:00
# {File.chmod}[rdoc-ref:File.chmod].
2022-06-09 23:03:21 +03:00
# - Modifies each entry that is a symbolic link using
2022-10-12 05:54:16 +03:00
# {File.lchmod}[rdoc-ref:File.lchmod].
2022-06-09 23:03:21 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
2022-06-09 23:03:21 +03:00
#
# Argument +mode+ may be either an integer or a string:
#
# - \Integer +mode+: represents the permission bits to be set:
#
# FileUtils.chmod(0755, 'src0.txt')
# FileUtils.chmod(0644, ['src0.txt', 'src0.dat'])
#
# - \String +mode+: represents the permissions to be set:
#
# The string is of the form <tt>[targets][[operator][perms[,perms]]</tt>, where:
#
# - +targets+ may be any combination of these letters:
#
# - <tt>'u'</tt>: permissions apply to the file's owner.
# - <tt>'g'</tt>: permissions apply to users in the file's group.
# - <tt>'o'</tt>: permissions apply to other users not in the file's group.
# - <tt>'a'</tt> (the default): permissions apply to all users.
#
# - +operator+ may be one of these letters:
#
# - <tt>'+'</tt>: adds permissions.
# - <tt>'-'</tt>: removes permissions.
# - <tt>'='</tt>: sets (replaces) permissions.
#
# - +perms+ (may be repeated, with separating commas)
# may be any combination of these letters:
#
# - <tt>'r'</tt>: Read.
# - <tt>'w'</tt>: Write.
# - <tt>'x'</tt>: Execute (search, for a directory).
# - <tt>'X'</tt>: Search (for a directories only;
# must be used with <tt>'+'</tt>)
# - <tt>'s'</tt>: Uid or gid.
# - <tt>'t'</tt>: Sticky bit.
#
# Examples:
#
# FileUtils.chmod('u=wrx,go=rx', 'src1.txt')
# FileUtils.chmod('u=wrx,go=rx', '/usr/bin/ruby')
2022-06-11 00:41:13 +03:00
#
2022-06-09 23:03:21 +03:00
# Keyword arguments:
#
# - <tt>noop: true</tt> - does not change permissions; returns +nil+.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.chmod(0755, 'src0.txt', noop: true, verbose: true)
# FileUtils.chmod(0644, ['src0.txt', 'src0.dat'], noop: true, verbose: true)
# FileUtils.chmod('u=wrx,go=rx', 'src1.txt', noop: true, verbose: true)
# FileUtils.chmod('u=wrx,go=rx', '/usr/bin/ruby', noop: true, verbose: true)
#
# Output:
#
# chmod 755 src0.txt
# chmod 644 src0.txt src0.dat
# chmod u=wrx,go=rx src1.txt
# chmod u=wrx,go=rx /usr/bin/ruby
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.chmod_R.
#
2016-03-01 05:26:44 +03:00
def chmod ( mode , list , noop : nil , verbose : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
fu_output_message sprintf ( 'chmod %s %s' , mode_to_s ( mode ) , list . join ( ' ' ) ) if verbose
return if noop
2005-05-26 02:41:32 +04:00
list . each do | path |
2011-05-01 20:16:05 +04:00
Entry_ . new ( path ) . chmod ( fu_mode ( mode , path ) )
2005-05-26 02:41:32 +04:00
end
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :chmod
2002-03-26 05:03:04 +03:00
2022-06-11 00:41:13 +03:00
# Like FileUtils.chmod, but changes permissions recursively.
2009-03-06 06:56:38 +03:00
#
2016-03-01 05:26:44 +03:00
def chmod_R ( mode , list , noop : nil , verbose : nil , force : nil )
2005-05-21 21:40:24 +04:00
list = fu_list ( list )
2013-02-02 07:07:17 +04:00
fu_output_message sprintf ( 'chmod -R%s %s %s' ,
2016-03-01 05:26:44 +03:00
( force ? 'f' : '' ) ,
mode_to_s ( mode ) , list . join ( ' ' ) ) if verbose
return if noop
2005-05-26 02:41:32 +04:00
list . each do | root |
Entry_ . new ( root ) . traverse do | ent |
begin
2011-05-01 20:16:05 +04:00
ent . chmod ( fu_mode ( mode , ent . path ) )
2005-05-26 02:41:32 +04:00
rescue
2016-03-01 05:26:44 +03:00
raise unless force
2005-05-21 21:40:24 +04:00
end
end
end
end
2013-03-01 06:09:42 +04:00
module_function :chmod_R
2005-05-21 21:40:24 +04:00
2022-06-11 00:41:13 +03:00
# Changes the owner and group on the entries at the paths given in +list+
2022-06-14 16:52:18 +03:00
# (a single path or an array of paths)
2022-06-16 17:56:33 +03:00
# to the given +user+ and +group+;
# returns +list+ if it is an array, <tt>[list]</tt> otherwise:
2022-06-11 00:41:13 +03:00
#
# - Modifies each entry that is a regular file using
2022-10-12 05:54:16 +03:00
# {File.chown}[rdoc-ref:File.chown].
2022-06-11 00:41:13 +03:00
# - Modifies each entry that is a symbolic link using
2022-10-12 05:54:16 +03:00
# {File.lchown}[rdoc-ref:File.lchown].
2022-06-11 00:41:13 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
2022-06-11 00:41:13 +03:00
#
# User and group:
#
# - Argument +user+ may be a user name or a user id;
# if +nil+ or +-1+, the user is not changed.
# - Argument +group+ may be a group name or a group id;
# if +nil+ or +-1+, the group is not changed.
# - The user must be a member of the group.
#
# Examples:
#
2022-06-13 18:39:41 +03:00
# # One path.
2022-06-11 00:41:13 +03:00
# # User and group as string names.
# File.stat('src0.txt').uid # => 1004
# File.stat('src0.txt').gid # => 1004
# FileUtils.chown('user2', 'group1', 'src0.txt')
# File.stat('src0.txt').uid # => 1006
# File.stat('src0.txt').gid # => 1005
2009-03-06 06:56:38 +03:00
#
2022-06-11 00:41:13 +03:00
# # User and group as uid and gid.
# FileUtils.chown(1004, 1004, 'src0.txt')
# File.stat('src0.txt').uid # => 1004
# File.stat('src0.txt').gid # => 1004
#
2022-06-13 18:39:41 +03:00
# # Array of paths.
2022-06-11 00:41:13 +03:00
# FileUtils.chown(1006, 1005, ['src0.txt', 'src0.dat'])
#
# # Directory (not recursive).
# FileUtils.chown('user2', 'group1', '.')
#
# Keyword arguments:
#
# - <tt>noop: true</tt> - does not change permissions; returns +nil+.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.chown('user2', 'group1', 'src0.txt', noop: true, verbose: true)
# FileUtils.chown(1004, 1004, 'src0.txt', noop: true, verbose: true)
# FileUtils.chown(1006, 1005, ['src0.txt', 'src0.dat'], noop: true, verbose: true)
# FileUtils.chown('user2', 'group1', path, noop: true, verbose: true)
# FileUtils.chown('user2', 'group1', '.', noop: true, verbose: true)
#
# Output:
2009-03-06 06:56:38 +03:00
#
2022-06-11 00:41:13 +03:00
# chown user2:group1 src0.txt
# chown 1004:1004 src0.txt
# chown 1006:1005 src0.txt src0.dat
# chown user2:group1 src0.txt
# chown user2:group1 .
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.chown_R.
#
2016-03-01 05:26:44 +03:00
def chown ( user , group , list , noop : nil , verbose : nil )
2004-12-04 14:46:24 +03:00
list = fu_list ( list )
2013-07-09 11:01:59 +04:00
fu_output_message sprintf ( 'chown %s %s' ,
( group ? " #{ user } : #{ group } " : user || ':' ) ,
2016-03-01 05:26:44 +03:00
list . join ( ' ' ) ) if verbose
return if noop
2005-05-26 02:41:32 +04:00
uid = fu_get_uid ( user )
gid = fu_get_gid ( group )
list . each do | path |
Entry_ . new ( path ) . chown uid , gid
end
2004-12-04 14:46:24 +03:00
end
2013-03-01 06:09:42 +04:00
module_function :chown
2004-12-04 14:46:24 +03:00
2022-06-13 15:11:32 +03:00
# Like FileUtils.chown, but changes owner and group recursively.
2009-03-06 06:56:38 +03:00
#
2016-03-01 05:26:44 +03:00
def chown_R ( user , group , list , noop : nil , verbose : nil , force : nil )
2004-12-04 14:46:24 +03:00
list = fu_list ( list )
2013-07-09 11:01:59 +04:00
fu_output_message sprintf ( 'chown -R%s %s %s' ,
2016-03-01 05:26:44 +03:00
( force ? 'f' : '' ) ,
2013-07-09 11:01:59 +04:00
( group ? " #{ user } : #{ group } " : user || ':' ) ,
2016-03-01 05:26:44 +03:00
list . join ( ' ' ) ) if verbose
return if noop
2004-12-04 14:46:24 +03:00
uid = fu_get_uid ( user )
gid = fu_get_gid ( group )
2005-05-26 02:41:32 +04:00
list . each do | root |
Entry_ . new ( root ) . traverse do | ent |
begin
ent . chown uid , gid
rescue
2016-03-01 05:26:44 +03:00
raise unless force
2005-05-21 21:40:24 +04:00
end
2004-12-04 14:46:24 +03:00
end
end
end
2013-03-01 06:09:42 +04:00
module_function :chown_R
2004-12-04 14:46:24 +03:00
2014-04-29 07:51:13 +04:00
def fu_get_uid ( user ) #:nodoc:
return nil unless user
case user
when Integer
user
when / \ A \ d+ \ z /
user . to_i
else
2019-03-27 11:26:09 +03:00
require 'etc'
2014-04-29 07:51:13 +04:00
Etc . getpwnam ( user ) ? Etc . getpwnam ( user ) . uid : nil
2004-12-04 14:46:24 +03:00
end
2014-04-29 07:51:13 +04:00
end
private_module_function :fu_get_uid
def fu_get_gid ( group ) #:nodoc:
return nil unless group
case group
when Integer
group
when / \ A \ d+ \ z /
group . to_i
else
2019-03-27 11:26:09 +03:00
require 'etc'
2014-04-29 07:51:13 +04:00
Etc . getgrnam ( group ) ? Etc . getgrnam ( group ) . gid : nil
2004-12-04 14:46:24 +03:00
end
end
2014-04-29 07:51:13 +04:00
private_module_function :fu_get_gid
2004-12-04 14:46:24 +03:00
2022-06-13 15:11:32 +03:00
# Updates modification times (mtime) and access times (atime)
2022-06-14 16:52:18 +03:00
# of the entries given by the paths in +list+
# (a single path or an array of paths);
2022-06-16 17:56:33 +03:00
# returns +list+ if it is an array, <tt>[list]</tt> otherwise.
#
# By default, creates an empty file for any path to a non-existent entry;
# use keyword argument +nocreate+ to raise an exception instead.
2009-03-06 06:56:38 +03:00
#
2022-06-13 18:39:41 +03:00
# Argument +list+ or its elements
# should be {interpretable as paths}[rdoc-ref:FileUtils@Path+Arguments].
#
# Examples:
#
# # Single path.
2022-06-13 15:11:32 +03:00
# f = File.new('src0.txt') # Existing file.
# f.atime # => 2022-06-10 11:11:21.200277 -0700
# f.mtime # => 2022-06-10 11:11:21.200277 -0700
# FileUtils.touch('src0.txt')
# f = File.new('src0.txt')
# f.atime # => 2022-06-11 08:28:09.8185343 -0700
# f.mtime # => 2022-06-11 08:28:09.8185343 -0700
2009-03-06 06:56:38 +03:00
#
2022-06-13 18:39:41 +03:00
# # Array of paths.
2022-06-13 15:11:32 +03:00
# FileUtils.touch(['src0.txt', 'src0.dat'])
#
# Keyword arguments:
#
# - <tt>mtime: <i>time</i></tt> - sets the entry's mtime to the given time,
# instead of the current time.
# - <tt>nocreate: true</tt> - raises an exception if the entry does not exist.
# - <tt>noop: true</tt> - does not touch entries; returns +nil+.
# - <tt>verbose: true</tt> - prints an equivalent command:
#
# FileUtils.touch('src0.txt', noop: true, verbose: true)
# FileUtils.touch(['src0.txt', 'src0.dat'], noop: true, verbose: true)
# FileUtils.touch(path, noop: true, verbose: true)
#
# Output:
#
# touch src0.txt
# touch src0.txt src0.dat
# touch src0.txt
2009-03-06 06:56:38 +03:00
#
2022-06-21 16:42:27 +03:00
# Related: FileUtils.uptodate?.
#
2016-03-01 05:26:44 +03:00
def touch ( list , noop : nil , verbose : nil , mtime : nil , nocreate : nil )
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2016-03-01 05:26:44 +03:00
t = mtime
if verbose
2011-04-15 10:05:25 +04:00
fu_output_message " touch #{ nocreate ? '-c ' : '' } #{ t ? t . strftime ( '-t %Y%m%d%H%M.%S ' ) : '' } #{ list . join ' ' } "
2007-03-03 16:37:02 +03:00
end
2016-03-01 05:26:44 +03:00
return if noop
2005-05-21 21:40:24 +04:00
list . each do | path |
2007-03-03 16:37:02 +03:00
created = nocreate
2002-03-26 05:03:04 +03:00
begin
2005-05-21 21:40:24 +04:00
File . utime ( t , t , path )
2002-03-26 05:03:04 +03:00
rescue Errno :: ENOENT
2007-03-03 16:37:02 +03:00
raise if created
2005-05-21 21:40:24 +04:00
File . open ( path , 'a' ) {
2003-12-12 13:33:24 +03:00
;
2003-05-02 10:07:54 +04:00
}
2007-03-03 16:37:02 +03:00
created = true
retry if t
2002-03-26 05:03:04 +03:00
end
end
end
2013-03-01 06:09:42 +04:00
module_function :touch
2002-03-26 05:03:04 +03:00
2013-03-01 06:09:42 +04:00
private
2002-03-26 05:03:04 +03:00
2023-12-06 04:37:14 +03:00
module StreamUtils_ # :nodoc:
2005-05-26 02:41:32 +04:00
private
2018-05-19 03:22:39 +03:00
case ( defined? ( :: RbConfig ) ? :: RbConfig :: CONFIG [ 'host_os' ] : :: RUBY_PLATFORM )
when / mswin|mingw /
2023-12-06 04:37:14 +03:00
def fu_windows? ; true end #:nodoc:
2018-05-19 03:22:39 +03:00
else
2023-12-06 04:37:14 +03:00
def fu_windows? ; false end #:nodoc:
2005-05-26 02:41:32 +04:00
end
2008-09-01 06:31:56 +04:00
def fu_copy_stream0 ( src , dest , blksize = nil ) #:nodoc:
IO . copy_stream ( src , dest )
2005-05-21 21:40:24 +04:00
end
2023-12-06 04:37:14 +03:00
def fu_stream_blksize ( * streams ) #:nodoc:
2005-05-26 02:41:32 +04:00
streams . each do | s |
next unless s . respond_to? ( :stat )
size = fu_blksize ( s . stat )
return size if size
2005-05-21 21:40:24 +04:00
end
2005-05-26 02:41:32 +04:00
fu_default_blksize ( )
2005-05-21 21:40:24 +04:00
end
2023-12-06 04:37:14 +03:00
def fu_blksize ( st ) #:nodoc:
2005-05-26 02:41:32 +04:00
s = st . blksize
return nil unless s
return nil if s == 0
s
end
2005-05-21 21:40:24 +04:00
2023-12-06 04:37:14 +03:00
def fu_default_blksize #:nodoc:
2005-05-26 02:41:32 +04:00
1024
2002-03-26 05:03:04 +03:00
end
end
2005-05-26 02:41:32 +04:00
include StreamUtils_
2005-09-19 00:59:29 +04:00
extend StreamUtils_
2005-05-26 02:41:32 +04:00
class Entry_ #:nodoc: internal use only
include StreamUtils_
def initialize ( a , b = nil , deref = false )
@prefix = @rel = @path = nil
if b
@prefix = a
@rel = b
else
@path = a
end
@deref = deref
@stat = nil
@lstat = nil
end
def inspect
" \# < #{ self . class } #{ path ( ) } > "
end
def path
if @path
2005-09-24 03:30:09 +04:00
File . path ( @path )
2005-05-26 02:41:32 +04:00
else
join ( @prefix , @rel )
end
end
def prefix
@prefix || @path
end
def rel
@rel
end
def dereference?
@deref
end
def exist?
2014-07-11 11:51:19 +04:00
begin
lstat
true
rescue Errno :: ENOENT
false
end
2005-05-26 02:41:32 +04:00
end
def file?
s = lstat!
s and s . file?
end
def directory?
s = lstat!
s and s . directory?
end
def symlink?
s = lstat!
s and s . symlink?
end
def chardev?
s = lstat!
s and s . chardev?
end
def blockdev?
s = lstat!
s and s . blockdev?
end
def socket?
s = lstat!
s and s . socket?
end
def pipe?
s = lstat!
s and s . pipe?
end
S_IF_DOOR = 0xD000
def door?
s = lstat!
s and ( s . mode & 0xF000 == S_IF_DOOR )
end
def entries
2010-05-17 04:41:56 +04:00
opts = { }
2020-06-22 12:15:21 +03:00
opts [ :encoding ] = fu_windows? ? :: Encoding :: UTF_8 : path . encoding
2019-03-27 11:26:09 +03:00
2021-01-17 05:52:03 +03:00
files = Dir . children ( path , ** opts )
2019-03-27 11:26:09 +03:00
2019-10-18 19:36:11 +03:00
untaint = RUBY_VERSION < '2.7'
files . map { | n | Entry_ . new ( prefix ( ) , join ( rel ( ) , untaint ? n . untaint : n ) ) }
2005-05-26 02:41:32 +04:00
end
def stat
return @stat if @stat
if lstat ( ) and lstat ( ) . symlink?
@stat = File . stat ( path ( ) )
else
@stat = lstat ( )
end
@stat
end
def stat!
return @stat if @stat
if lstat! and lstat! . symlink?
@stat = File . stat ( path ( ) )
else
@stat = lstat!
end
@stat
rescue SystemCallError
nil
end
def lstat
if dereference?
@lstat || = File . stat ( path ( ) )
else
@lstat || = File . lstat ( path ( ) )
end
end
def lstat!
lstat ( )
rescue SystemCallError
nil
end
def chmod ( mode )
if symlink?
File . lchmod mode , path ( ) if have_lchmod?
else
File . chmod mode , path ( )
end
2020-01-23 09:33:42 +03:00
rescue Errno :: EOPNOTSUPP
2005-05-26 02:41:32 +04:00
end
def chown ( uid , gid )
if symlink?
File . lchown uid , gid , path ( ) if have_lchown?
else
File . chown uid , gid , path ( )
end
end
2018-03-13 11:18:03 +03:00
def link ( dest )
case
when directory?
if ! File . exist? ( dest ) and descendant_directory? ( dest , path )
raise ArgumentError , " cannot link directory %s to itself %s " % [ path , dest ]
end
begin
Dir . mkdir dest
rescue
raise unless File . directory? ( dest )
end
else
File . link path ( ) , dest
end
end
2005-05-26 02:41:32 +04:00
def copy ( dest )
2016-11-04 04:35:49 +03:00
lstat
2005-05-26 02:41:32 +04:00
case
when file?
copy_file dest
when directory?
2014-08-12 07:21:57 +04:00
if ! File . exist? ( dest ) and descendant_directory? ( dest , path )
2007-09-06 17:59:37 +04:00
raise ArgumentError , " cannot copy directory %s to itself %s " % [ path , dest ]
end
2005-05-26 02:41:32 +04:00
begin
Dir . mkdir dest
rescue
raise unless File . directory? ( dest )
end
when symlink?
File . symlink File . readlink ( path ( ) ) , dest
2019-07-10 06:41:51 +03:00
when chardev? , blockdev?
raise " cannot handle device file "
2005-05-26 02:41:32 +04:00
when socket?
2019-07-10 06:41:51 +03:00
begin
require 'socket'
rescue LoadError
raise " cannot handle socket "
else
raise " cannot handle socket " unless defined? ( UNIXServer )
end
UNIXServer . new ( dest ) . close
File . chmod lstat ( ) . mode , dest
2005-05-26 02:41:32 +04:00
when pipe?
raise " cannot handle FIFO " unless File . respond_to? ( :mkfifo )
2019-07-10 06:41:51 +03:00
File . mkfifo dest , lstat ( ) . mode
2005-05-26 02:41:32 +04:00
when door?
raise " cannot handle door: #{ path ( ) } "
else
raise " unknown file type: #{ path ( ) } "
end
end
def copy_file ( dest )
2009-09-09 17:02:04 +04:00
File . open ( path ( ) ) do | s |
2011-03-18 02:48:13 +03:00
File . open ( dest , 'wb' , s . stat . mode ) do | f |
2009-09-09 17:02:04 +04:00
IO . copy_stream ( s , f )
end
2009-06-21 05:47:25 +04:00
end
2005-05-26 02:41:32 +04:00
end
def copy_metadata ( path )
st = lstat ( )
2012-04-30 18:37:54 +04:00
if ! st . symlink?
File . utime st . atime , st . mtime , path
end
2016-06-27 04:24:46 +03:00
mode = st . mode
2005-05-26 02:41:32 +04:00
begin
2012-04-30 18:37:54 +04:00
if st . symlink?
begin
File . lchown st . uid , st . gid , path
rescue NotImplementedError
end
else
File . chown st . uid , st . gid , path
end
2016-06-28 16:14:48 +03:00
rescue Errno :: EPERM , Errno :: EACCES
2005-05-26 02:41:32 +04:00
# clear setuid/setgid
2016-06-27 04:24:46 +03:00
mode & = 01777
end
if st . symlink?
begin
File . lchmod mode , path
2020-01-23 09:33:42 +03:00
rescue NotImplementedError , Errno :: EOPNOTSUPP
2012-04-30 18:37:54 +04:00
end
2005-05-26 02:41:32 +04:00
else
2016-06-27 04:24:46 +03:00
File . chmod mode , path
2005-05-26 02:41:32 +04:00
end
end
def remove
if directory?
remove_dir1
else
remove_file
end
end
def remove_dir1
platform_support {
2012-02-25 12:08:55 +04:00
Dir . rmdir path ( ) . chomp ( ?/ )
2005-05-26 02:41:32 +04:00
}
end
def remove_file
platform_support {
File . unlink path
}
end
def platform_support
return yield unless fu_windows?
first_time_p = true
begin
yield
rescue Errno :: ENOENT
raise
rescue = > err
if first_time_p
first_time_p = false
begin
File . chmod 0700 , path ( ) # Windows does not have symlink
retry
rescue SystemCallError
end
end
raise err
end
end
def preorder_traverse
stack = [ self ]
while ent = stack . pop
yield ent
stack . concat ent . entries . reverse if ent . directory?
end
end
alias traverse preorder_traverse
def postorder_traverse
if directory?
2022-07-26 15:23:47 +03:00
begin
children = entries ( )
rescue Errno :: EACCES
# Failed to get the list of children.
# Assuming there is no children, try to process the parent directory.
yield self
return
end
children . each do | ent |
2005-05-26 02:41:32 +04:00
ent . postorder_traverse do | e |
yield e
end
end
end
yield self
end
2013-02-02 07:54:00 +04:00
def wrap_traverse ( pre , post )
pre . call self
if directory?
entries . each do | ent |
ent . wrap_traverse pre , post
end
end
post . call self
end
2013-03-01 06:09:42 +04:00
private
2005-05-26 02:41:32 +04:00
2019-08-24 15:19:31 +03:00
@@fileutils_rb_have_lchmod = nil
2005-05-26 02:41:32 +04:00
def have_lchmod?
# This is not MT-safe, but it does not matter.
2019-08-24 15:19:31 +03:00
if @@fileutils_rb_have_lchmod == nil
@@fileutils_rb_have_lchmod = check_have_lchmod?
2005-05-26 02:41:32 +04:00
end
2019-08-24 15:19:31 +03:00
@@fileutils_rb_have_lchmod
2005-05-26 02:41:32 +04:00
end
def check_have_lchmod?
return false unless File . respond_to? ( :lchmod )
File . lchmod 0
return true
rescue NotImplementedError
return false
end
2019-08-24 15:19:31 +03:00
@@fileutils_rb_have_lchown = nil
2005-05-26 02:41:32 +04:00
def have_lchown?
# This is not MT-safe, but it does not matter.
2019-08-24 15:19:31 +03:00
if @@fileutils_rb_have_lchown == nil
@@fileutils_rb_have_lchown = check_have_lchown?
2005-05-26 02:41:32 +04:00
end
2019-08-24 15:19:31 +03:00
@@fileutils_rb_have_lchown
2005-05-26 02:41:32 +04:00
end
def check_have_lchown?
return false unless File . respond_to? ( :lchown )
File . lchown nil , nil
return true
rescue NotImplementedError
return false
end
def join ( dir , base )
return File . path ( dir ) if not base or base == '.'
return File . path ( base ) if not dir or dir == '.'
2020-06-22 12:34:24 +03:00
begin
File . join ( dir , base )
rescue EncodingError
if fu_windows?
File . join ( dir . encode ( :: Encoding :: UTF_8 ) , base . encode ( :: Encoding :: UTF_8 ) )
else
raise
end
end
2005-05-26 02:41:32 +04:00
end
2010-07-20 07:27:59 +04:00
if File :: ALT_SEPARATOR
2017-01-07 05:14:07 +03:00
DIRECTORY_TERM = " (?=[/ #{ Regexp . quote ( File :: ALT_SEPARATOR ) } ]| \\ z) "
2010-07-20 07:27:59 +04:00
else
2017-01-07 05:14:07 +03:00
DIRECTORY_TERM = " (?=/| \\ z) "
2010-07-20 07:27:59 +04:00
end
2014-08-12 07:21:57 +04:00
def descendant_directory? ( descendant , ascendant )
2016-09-11 12:09:43 +03:00
if File :: FNM_SYSCASE . nonzero?
File . expand_path ( File . dirname ( descendant ) ) . casecmp ( File . expand_path ( ascendant ) ) == 0
else
File . expand_path ( File . dirname ( descendant ) ) == File . expand_path ( ascendant )
end
2010-07-20 07:27:59 +04:00
end
2005-05-26 02:41:32 +04:00
end # class Entry_
2005-09-19 00:59:29 +04:00
def fu_list ( arg ) #:nodoc:
2004-05-07 16:04:19 +04:00
[ arg ] . flatten . map { | path | File . path ( path ) }
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_list
2002-03-26 05:03:04 +03:00
2005-09-19 00:59:29 +04:00
def fu_each_src_dest ( src , dest ) #:nodoc:
2003-11-18 08:09:20 +03:00
fu_each_src_dest0 ( src , dest ) do | s , d |
raise ArgumentError , " same file: #{ s } and #{ d } " if fu_same? ( s , d )
2014-07-11 11:51:19 +04:00
yield s , d
2003-11-18 08:09:20 +03:00
end
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_each_src_dest
2003-11-18 08:09:20 +03:00
2022-11-25 04:03:51 +03:00
def fu_each_src_dest0 ( src , dest , target_directory = true ) #:nodoc:
2007-09-06 17:59:37 +04:00
if tmp = Array . try_convert ( src )
tmp . each do | s |
2004-04-07 06:51:05 +04:00
s = File . path ( s )
2022-11-25 04:03:51 +03:00
yield s , ( target_directory ? File . join ( dest , File . basename ( s ) ) : dest )
2002-03-26 05:03:04 +03:00
end
else
2004-04-07 06:51:05 +04:00
src = File . path ( src )
2022-11-25 04:03:51 +03:00
if target_directory and File . directory? ( dest )
2004-04-07 06:51:05 +04:00
yield src , File . join ( dest , File . basename ( src ) )
2003-12-12 13:33:24 +03:00
else
2004-04-07 06:51:05 +04:00
yield src , File . path ( dest )
2003-12-12 13:33:24 +03:00
end
2002-03-26 05:03:04 +03:00
end
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_each_src_dest0
2002-03-26 05:03:04 +03:00
2005-09-19 00:59:29 +04:00
def fu_same? ( a , b ) #:nodoc:
2010-04-12 04:19:52 +04:00
File . identical? ( a , b )
2003-11-18 08:09:20 +03:00
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_same?
2003-11-18 08:09:20 +03:00
2005-09-19 00:59:29 +04:00
def fu_output_message ( msg ) #:nodoc:
2019-07-31 22:57:21 +03:00
output = @fileutils_output if defined? ( @fileutils_output )
2020-12-19 10:53:18 +03:00
output || = $stdout
2019-07-31 22:57:21 +03:00
if defined? ( @fileutils_label )
msg = @fileutils_label + msg
end
output . puts msg
2002-03-26 05:03:04 +03:00
end
2013-03-01 06:09:42 +04:00
private_module_function :fu_output_message
2002-03-26 05:03:04 +03:00
2023-12-06 04:37:14 +03:00
def fu_split_path ( path ) #:nodoc:
2022-11-25 04:03:51 +03:00
path = File . path ( path )
list = [ ]
until ( parent , base = File . split ( path ) ; parent == path or parent == " . " )
list << base
path = parent
end
list << path
list . reverse!
end
private_module_function :fu_split_path
def fu_relative_components_from ( target , base ) #:nodoc:
i = 0
while target [ i ] & . == base [ i ]
i += 1
end
Array . new ( base . size - i , '..' ) . concat ( target [ i .. - 1 ] )
end
private_module_function :fu_relative_components_from
2023-12-06 04:37:14 +03:00
def fu_clean_components ( * comp ) #:nodoc:
2022-11-25 04:03:51 +03:00
comp . shift while comp . first == " . "
return comp if comp . empty?
clean = [ comp . shift ]
path = File . join ( * clean , " " ) # ending with File::SEPARATOR
while c = comp . shift
if c == " .. " and clean . last != " .. " and ! ( fu_have_symlink? && File . symlink? ( path ) )
clean . pop
path . chomp! ( %r( ( ?<= \ A|/ ) [^/]+/ \ z ) , " " )
else
clean << c
path << c << " / "
end
end
clean
end
private_module_function :fu_clean_components
if fu_windows?
2023-12-06 04:37:14 +03:00
def fu_starting_path? ( path ) #:nodoc:
2022-11-25 04:03:51 +03:00
path & . start_with? ( %r( \ w:|/ ) )
end
else
2023-12-06 04:37:14 +03:00
def fu_starting_path? ( path ) #:nodoc:
2022-11-25 04:03:51 +03:00
path & . start_with? ( " / " )
end
end
private_module_function :fu_starting_path?
2016-03-01 05:26:44 +03:00
# This hash table holds command options.
OPT_TABLE = { } #:nodoc: internal use only
( private_instance_methods & methods ( false ) ) . inject ( OPT_TABLE ) { | tbl , name |
( tbl [ name . to_s ] = instance_method ( name ) . parameters ) . map! { | t , n | n if t == :key } . compact!
tbl
}
2019-03-10 14:09:59 +03:00
public
2022-06-13 15:11:32 +03:00
# Returns an array of the string names of \FileUtils methods
2022-06-17 16:36:20 +03:00
# that accept one or more keyword arguments:
2004-05-07 17:08:12 +04:00
#
2022-06-13 15:11:32 +03:00
# FileUtils.commands.sort.take(3) # => ["cd", "chdir", "chmod"]
2004-05-07 17:08:12 +04:00
#
2016-02-29 11:36:28 +03:00
def self . commands
2004-05-07 17:08:12 +04:00
OPT_TABLE . keys
end
2022-06-13 15:11:32 +03:00
# Returns an array of the string keyword names:
2004-05-07 17:08:12 +04:00
#
2022-06-13 15:11:32 +03:00
# FileUtils.options.take(3) # => ["noop", "verbose", "force"]
2004-05-07 17:08:12 +04:00
#
2016-02-29 11:36:28 +03:00
def self . options
2006-03-05 12:41:33 +03:00
OPT_TABLE . values . flatten . uniq . map { | sym | sym . to_s }
2004-05-07 17:08:12 +04:00
end
2022-06-13 15:11:32 +03:00
# Returns +true+ if method +mid+ accepts the given option +opt+, +false+ otherwise;
# the arguments may be strings or symbols:
2004-05-07 17:08:12 +04:00
#
2022-06-13 15:11:32 +03:00
# FileUtils.have_option?(:chmod, :noop) # => true
# FileUtils.have_option?('chmod', 'secure') # => false
2004-05-07 17:08:12 +04:00
#
2016-02-29 11:36:28 +03:00
def self . have_option? ( mid , opt )
2005-05-26 02:41:32 +04:00
li = OPT_TABLE [ mid . to_s ] or raise ArgumentError , " no such method: #{ mid } "
2006-03-05 12:41:33 +03:00
li . include? ( opt )
2004-05-07 17:08:12 +04:00
end
2022-06-13 15:11:32 +03:00
# Returns an array of the string keyword name for method +mid+;
# the argument may be a string or a symbol:
2004-05-07 17:08:12 +04:00
#
2022-06-13 15:11:32 +03:00
# FileUtils.options_of(:rm) # => ["force", "noop", "verbose"]
# FileUtils.options_of('mv') # => ["force", "noop", "verbose", "secure"]
2004-05-07 17:08:12 +04:00
#
2016-02-29 11:36:28 +03:00
def self . options_of ( mid )
2006-03-05 12:41:33 +03:00
OPT_TABLE [ mid . to_s ] . map { | sym | sym . to_s }
2004-05-07 17:08:12 +04:00
end
2022-06-13 15:11:32 +03:00
# Returns an array of the string method names of the methods
# that accept the given keyword option +opt+;
# the argument must be a symbol:
2004-05-07 17:08:12 +04:00
#
2022-06-13 15:11:32 +03:00
# FileUtils.collect_method(:preserve) # => ["cp", "copy", "cp_r", "install"]
2004-05-07 17:08:12 +04:00
#
2016-02-29 11:36:28 +03:00
def self . collect_method ( opt )
2006-03-05 12:41:33 +03:00
OPT_TABLE . keys . select { | m | OPT_TABLE [ m ] . include? ( opt ) }
2004-05-07 17:08:12 +04:00
end
2019-03-10 14:09:59 +03:00
private
LOW_METHODS = singleton_methods ( false ) - collect_method ( :noop ) . map ( & :intern ) # :nodoc:
module LowMethods # :nodoc: internal use only
2016-02-29 15:22:17 +03:00
private
def _do_nothing ( * ) end
:: FileUtils :: LOW_METHODS . map { | name | alias_method name , :_do_nothing }
2013-03-01 06:09:42 +04:00
end
2019-03-10 14:09:59 +03:00
METHODS = singleton_methods ( ) - [ :private_module_function , # :nodoc:
2013-03-01 06:09:42 +04:00
:commands , :options , :have_option? , :options_of , :collect_method ]
2009-03-06 06:56:38 +03:00
#
2013-03-01 06:09:42 +04:00
# This module has all methods of FileUtils module, but it outputs messages
# before acting. This equates to passing the <tt>:verbose</tt> flag to
# methods in FileUtils.
2009-03-06 06:56:38 +03:00
#
2013-03-01 06:09:42 +04:00
module Verbose
include FileUtils
2016-02-29 11:36:30 +03:00
names = :: FileUtils . collect_method ( :verbose )
names . each do | name |
2013-03-01 06:09:42 +04:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
2016-03-01 05:26:44 +03:00
def #{name}(*args, **options)
super ( * args , ** options , verbose : true )
2013-03-01 06:09:42 +04:00
end
EOS
end
2016-02-29 11:36:30 +03:00
private ( * names )
2013-03-01 06:09:42 +04:00
extend self
class << self
2016-02-29 11:36:29 +03:00
public ( * :: FileUtils :: METHODS )
2013-03-01 06:09:42 +04:00
end
end
2002-03-26 05:03:04 +03:00
2013-03-01 06:09:42 +04:00
#
# This module has all methods of FileUtils module, but never changes
# files/directories. This equates to passing the <tt>:noop</tt> flag
# to methods in FileUtils.
#
module NoWrite
include FileUtils
include LowMethods
2016-02-29 11:36:30 +03:00
names = :: FileUtils . collect_method ( :noop )
names . each do | name |
2013-03-01 06:09:42 +04:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
2016-03-01 05:26:44 +03:00
def #{name}(*args, **options)
super ( * args , ** options , noop : true )
2013-03-01 06:09:42 +04:00
end
EOS
end
2016-02-29 11:36:30 +03:00
private ( * names )
2013-03-01 06:09:42 +04:00
extend self
class << self
2016-02-29 11:36:29 +03:00
public ( * :: FileUtils :: METHODS )
2013-03-01 06:09:42 +04:00
end
end
2002-03-26 05:03:04 +03:00
2009-03-06 06:56:38 +03:00
#
2013-03-01 06:09:42 +04:00
# This module has all methods of FileUtils module, but never changes
# files/directories, with printing message before acting.
# This equates to passing the <tt>:noop</tt> and <tt>:verbose</tt> flag
# to methods in FileUtils.
2009-03-06 06:56:38 +03:00
#
2013-03-01 06:09:42 +04:00
module DryRun
include FileUtils
include LowMethods
2016-02-29 11:36:30 +03:00
names = :: FileUtils . collect_method ( :noop )
names . each do | name |
2013-03-01 06:09:42 +04:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
2016-03-01 05:26:44 +03:00
def #{name}(*args, **options)
super ( * args , ** options , noop : true , verbose : true )
2013-03-01 06:09:42 +04:00
end
EOS
end
2016-02-29 11:36:30 +03:00
private ( * names )
2013-03-01 06:09:42 +04:00
extend self
class << self
2016-02-29 11:36:29 +03:00
public ( * :: FileUtils :: METHODS )
2013-03-01 06:09:42 +04:00
end
end
2004-02-16 19:23:07 +03:00
end