2003-01-24 19:13:33 +03:00
#
2003-02-06 12:56:13 +03:00
# = fileutils.rb
2003-01-24 19:13:33 +03:00
#
2005-03-26 18:57:16 +03:00
# Copyright (c) 2000-2005 Minero Aoki <aamine@loveruby.net>
2003-01-24 19:13:33 +03:00
#
# This program is free software.
# You can distribute/modify this program under the same terms of ruby.
#
# == module FileUtils
#
# Namespace for several file utility methods for copying, moving, removing, etc.
#
# === Module Functions
#
2003-12-11 12:49:47 +03:00
# cd(dir, options)
# cd(dir, options) {|dir| .... }
2003-02-21 14:57:31 +03:00
# pwd()
2003-12-11 12:49:47 +03:00
# mkdir(dir, options)
2005-05-26 02:41:32 +04:00
# mkdir(list, options)
2003-12-11 12:49:47 +03:00
# mkdir_p(dir, options)
2005-05-26 02:41:32 +04:00
# mkdir_p(list, options)
2003-12-11 12:49:47 +03:00
# rmdir(dir, options)
2005-05-26 02:41:32 +04:00
# rmdir(list, options)
2003-12-11 12:49:47 +03:00
# ln(old, new, options)
# ln(list, destdir, options)
# ln_s(old, new, options)
# ln_s(list, destdir, options)
# ln_sf(src, dest, options)
# cp(src, dest, options)
# cp(list, dir, options)
# cp_r(src, dest, options)
# cp_r(list, dir, options)
# mv(src, dest, options)
# mv(list, dir, options)
# rm(list, options)
# rm_r(list, options)
# rm_rf(list, options)
# install(src, dest, mode = <src's>, options)
# chmod(mode, list, options)
2005-05-26 02:41:32 +04:00
# chmod_R(mode, list, options)
# chown(user, group, list, options)
# chown_R(user, group, list, options)
2003-12-11 12:49:47 +03:00
# touch(list, options)
2003-01-24 19:13:33 +03:00
#
2003-05-02 10:07:54 +04:00
# The <tt>options</tt> parameter is a hash of options, taken from the list
2004-02-13 15:53:42 +03:00
# <tt>:force</tt>, <tt>:noop</tt>, <tt>:preserve</tt>, and <tt>:verbose</tt>.
# <tt>:noop</tt> means that no changes are made. The other two are obvious.
# Each method documents the options that it honours.
2003-01-24 19:13:33 +03:00
#
# All methods that have the concept of a "source" file or directory can take
# either one file or a list of files in that argument. See the method
# documentation for examples.
#
2005-05-26 02:41:32 +04:00
# There are some `low level' methods, which do not accept any option:
2003-02-21 14:57:31 +03:00
#
2004-08-09 01:35:11 +04:00
# copy_entry(src, dest, preserve = false, dereference = false)
# copy_file(src, dest, preserve = false, dereference = true)
2003-12-11 12:49:47 +03:00
# copy_stream(srcstream, deststream)
2005-05-26 02:41:32 +04:00
# remove_entry(path, force = false)
# remove_entry_secure(path, force = false)
2005-01-01 10:22:04 +03:00
# remove_file(path, force = false)
2004-05-07 17:08:12 +04:00
# compare_file(path_a, path_b)
2003-12-11 12:49:47 +03:00
# compare_stream(stream_a, stream_b)
2005-01-01 10:22:04 +03:00
# uptodate?(file, cmp_list)
2003-02-21 14:57:31 +03:00
#
2003-01-24 19:13:33 +03:00
# == module FileUtils::Verbose
#
2003-02-06 12:56:13 +03:00
# This module has all methods of FileUtils module, but it outputs messages
2004-02-13 15:53:42 +03:00
# before acting. This equates to passing the <tt>:verbose</tt> flag to methods
# in FileUtils.
2003-01-24 19:13:33 +03:00
#
# == module FileUtils::NoWrite
#
2003-02-06 12:56:13 +03:00
# This module has all methods of FileUtils module, but never changes
2004-02-13 15:53:42 +03:00
# files/directories. This equates to passing the <tt>:noop</tt> flag to methods
# in FileUtils.
2003-01-24 19:13:33 +03:00
#
2004-05-07 17:08:12 +04:00
# == module FileUtils::DryRun
#
# This module has all methods of FileUtils module, but never changes
# files/directories. This equates to passing the <tt>:noop</tt> and
# <tt>:verbose</tt> flags to methods in FileUtils.
#
2003-01-24 19:13:33 +03:00
2002-03-26 05:03:04 +03:00
module FileUtils
2003-01-24 19:13:33 +03:00
# All methods are module_function.
2005-05-21 21:40:24 +04:00
# This hash table holds command options.
OPT_TABLE = { } #:nodoc: internal use only
2003-02-09 01:01:53 +03:00
#
# Options: (none)
#
# Returns the name of the current directory.
#
def pwd
Dir . pwd
end
alias getwd pwd
2003-01-24 19:13:33 +03:00
#
2005-05-26 02:41:32 +04:00
# Options: verbose
2003-01-24 19:13:33 +03:00
#
# Changes the current directory to the directory +dir+.
#
# If this method is called with block, resumes to the old
# working directory after the block execution finished.
#
2003-05-02 10:07:54 +04:00
# FileUtils.cd('/', :verbose => true) # chdir and report it
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def cd ( dir , options = { } , & block ) # :yield: dir
2005-05-26 02:41:32 +04:00
fu_check_options options , :verbose
2003-05-02 10:07:54 +04:00
fu_output_message " cd #{ dir } " if options [ :verbose ]
2005-08-13 17:06:04 +04:00
Dir . chdir ( dir , & block ) unless options [ :noop ]
2003-05-02 10:07:54 +04:00
fu_output_message 'cd -' if options [ :verbose ] and block
2002-03-26 05:03:04 +03:00
end
alias chdir cd
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'cd' ] =
2005-05-26 02:41:32 +04:00
OPT_TABLE [ 'chdir' ] = %w( verbose )
2005-05-21 21:40:24 +04:00
2003-01-24 19:13:33 +03:00
#
2003-02-09 01:01:53 +03:00
# Options: (none)
2003-01-24 19:13:33 +03:00
#
# Returns true if +newer+ is newer than all +old_list+.
# Non-existent files are older than any file.
#
2003-05-02 10:07:54 +04:00
# FileUtils.uptodate?('hello.o', %w(hello.c hello.h)) or \
# system 'make hello.o'
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def uptodate? ( new , old_list , options = nil )
2003-05-02 10:07:54 +04:00
raise ArgumentError , 'uptodate? does not accept any option' if options
2002-03-26 05:03:04 +03:00
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
2003-01-24 19:13:33 +03:00
#
2003-05-02 10:07:54 +04:00
# Options: mode noop verbose
2003-01-24 19:13:33 +03:00
#
# Creates one or more directories.
#
# FileUtils.mkdir 'test'
# FileUtils.mkdir %w( tmp data )
2003-05-02 10:07:54 +04:00
# FileUtils.mkdir 'notexist', :noop => true # Does not really create.
# FileUtils.mkdir 'tmp', :mode => 0700
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def mkdir ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :mode , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " mkdir #{ options [ :mode ] ? ( '-m %03o ' % options [ :mode ] ) : '' } #{ list . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2002-03-26 05:03:04 +03:00
list . each do | dir |
2004-12-04 14:46:24 +03:00
fu_mkdir dir , options [ :mode ]
2002-03-26 05:03:04 +03:00
end
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'mkdir' ] = %w( noop verbose mode )
2003-01-24 19:13:33 +03:00
#
2003-05-02 10:07:54 +04:00
# Options: mode noop verbose
2003-01-24 19:13:33 +03:00
#
# Creates a directory and all its parent directories.
# For example,
#
2003-02-06 12:56:13 +03:00
# FileUtils.mkdir_p '/usr/local/lib/ruby'
2003-01-24 19:13:33 +03:00
#
2003-02-06 12:56:13 +03:00
# causes to make following directories, if it does not exist.
2003-01-24 19:13:33 +03:00
# * /usr
# * /usr/local
2003-02-06 12:56:13 +03:00
# * /usr/local/lib
# * /usr/local/lib/ruby
2003-01-24 19:13:33 +03:00
#
# You can pass several directories at a time in a list.
#
2003-12-11 12:49:47 +03:00
def mkdir_p ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :mode , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " mkdir -p #{ options [ :mode ] ? ( '-m %03o ' % options [ :mode ] ) : '' } #{ list . join ' ' } " if options [ :verbose ]
return * list if options [ :noop ]
2002-03-26 05:03:04 +03:00
2003-12-11 14:54:50 +03:00
list . map { | path | path . sub ( %r< / \ z > , '' ) } . each do | path |
2004-08-09 01:35:11 +04:00
# optimize for the most common case
begin
2004-12-04 14:46:24 +03:00
fu_mkdir path , options [ :mode ]
2004-08-09 01:35:11 +04:00
next
rescue SystemCallError
next if File . directory? ( path )
end
2002-03-26 05:03:04 +03:00
stack = [ ]
2003-12-11 13:15:54 +03:00
until path == stack . last # dirname("/")=="/", dirname("C:/")=="C:/"
stack . push path
path = File . dirname ( path )
2002-03-26 05:03:04 +03:00
end
2003-12-11 13:15:54 +03:00
stack . reverse_each do | path |
begin
2004-12-04 14:46:24 +03:00
fu_mkdir path , options [ :mode ]
2003-12-11 13:54:17 +03:00
rescue SystemCallError = > err
2003-12-11 13:15:54 +03:00
raise unless File . directory? ( path )
end
2002-03-26 05:03:04 +03:00
end
end
return * list
end
alias mkpath mkdir_p
alias makedirs mkdir_p
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'mkdir_p' ] =
OPT_TABLE [ 'mkpath' ] =
OPT_TABLE [ 'makedirs' ] = %w( noop verbose )
2004-12-04 14:46:24 +03:00
def fu_mkdir ( path , mode )
path = path . sub ( %r< / \ z > , '' )
if mode
Dir . mkdir path , mode
File . chmod mode , path
else
Dir . mkdir path
end
end
private :fu_mkdir
2003-01-24 19:13:33 +03:00
#
# Options: noop, verbose
#
# Removes one or more directories.
#
# FileUtils.rmdir 'somedir'
# FileUtils.rmdir %w(somedir anydir otherdir)
# # Does not really remove directory; outputs message.
2003-05-02 10:07:54 +04:00
# FileUtils.rmdir 'somedir', :verbose => true, :noop => true
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def rmdir ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " rmdir #{ list . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2002-03-26 05:03:04 +03:00
list . each do | dir |
2003-12-12 13:33:24 +03:00
Dir . rmdir dir . sub ( %r< / \ z > , '' )
2002-03-26 05:03:04 +03:00
end
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'rmdir' ] = %w( noop verbose )
2003-01-24 19:13:33 +03:00
#
# Options: force noop verbose
#
2005-05-26 15:33:16 +04:00
# <b><tt>ln(old, new, options = {})</tt></b>
2003-01-24 19:13:33 +03:00
#
# Creates a hard link +new+ which points to +old+.
# If +new+ already exists and it is a directory, creates a symbolic link +new/old+.
# If +new+ already exists and it is not a directory, raises Errno::EEXIST.
# But if :force option is set, overwrite +new+.
#
2003-05-02 10:07:54 +04:00
# FileUtils.ln 'gcc', 'cc', :verbose => true
2003-01-24 19:13:33 +03:00
# FileUtils.ln '/usr/bin/emacs21', '/usr/bin/emacs'
#
2005-05-26 15:33:16 +04:00
# <b><tt>ln(list, destdir, options = {})</tt></b>
2003-01-24 19:13:33 +03:00
#
# Creates several hard links in a directory, with each one pointing to the
# item in +list+. If +destdir+ is not a directory, raises Errno::ENOTDIR.
#
# include FileUtils
2005-05-26 15:33:16 +04:00
# cd '/sbin'
# FileUtils.ln %w(cp mv mkdir), '/bin' # Now /sbin/cp and /bin/cp are linked.
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def ln ( src , dest , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :force , :noop , :verbose
fu_output_message " ln #{ options [ :force ] ? ' -f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2003-11-18 08:09:20 +03:00
fu_each_src_dest0 ( src , dest ) do | s , d |
2003-05-02 10:07:54 +04:00
remove_file d , true if options [ :force ]
2002-03-26 05:03:04 +03:00
File . link s , d
end
end
alias link ln
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'ln' ] =
OPT_TABLE [ 'link' ] = %w( noop verbose force )
2003-01-24 19:13:33 +03:00
#
# Options: force noop verbose
#
2005-05-26 15:33:16 +04:00
# <b><tt>ln_s(old, new, options = {})</tt></b>
2003-01-24 19:13:33 +03:00
#
# Creates a symbolic link +new+ which points to +old+. If +new+ already
# exists and it is a directory, creates a symbolic link +new/old+. If +new+
# already exists and it is not a directory, raises Errno::EEXIST. But if
# :force option is set, overwrite +new+.
#
# FileUtils.ln_s '/usr/bin/ruby', '/usr/local/bin/ruby'
2003-05-02 10:07:54 +04:00
# FileUtils.ln_s 'verylongsourcefilename.c', 'c', :force => true
2003-01-24 19:13:33 +03:00
#
2005-05-26 15:33:16 +04:00
# <b><tt>ln_s(list, destdir, options = {})</tt></b>
2003-01-24 19:13:33 +03:00
#
# Creates several symbolic links in a directory, with each one pointing to the
# item in +list+. If +destdir+ is not a directory, raises Errno::ENOTDIR.
#
# If +destdir+ is not a directory, raises Errno::ENOTDIR.
#
# FileUtils.ln_s Dir.glob('bin/*.rb'), '/home/aamine/bin'
#
2003-12-11 12:49:47 +03:00
def ln_s ( src , dest , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :force , :noop , :verbose
fu_output_message " ln -s #{ options [ :force ] ? 'f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2003-11-18 08:09:20 +03:00
fu_each_src_dest0 ( src , dest ) do | s , d |
2003-05-02 10:07:54 +04:00
remove_file d , true if options [ :force ]
2002-03-26 05:03:04 +03:00
File . symlink s , d
end
end
alias symlink ln_s
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'ln_s' ] =
OPT_TABLE [ 'symlink' ] = %w( noop verbose force )
2003-01-24 19:13:33 +03:00
#
# Options: noop verbose
#
# Same as
# #ln_s(src, dest, :force)
#
2003-12-11 12:49:47 +03:00
def ln_sf ( src , dest , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :noop , :verbose
options = options . dup
options [ :force ] = true
ln_s src , dest , options
2002-03-26 05:03:04 +03:00
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'ln_sf' ] = %w( noop verbose )
2003-01-24 19:13:33 +03:00
#
# Options: preserve noop verbose
#
2005-05-26 02:41:32 +04:00
# Copies a file content +src+ to +dest+. If +dest+ is a directory,
# copies +src+ to +dest/src+.
2003-01-24 19:13:33 +03:00
#
# If +src+ is a list of files, then +dest+ must be a directory.
#
# FileUtils.cp 'eval.c', 'eval.c.org'
2003-02-06 12:56:13 +03:00
# FileUtils.cp %w(cgi.rb complex.rb date.rb), '/usr/lib/ruby/1.6'
2003-05-02 10:07:54 +04:00
# FileUtils.cp %w(cgi.rb complex.rb date.rb), '/usr/lib/ruby/1.6', :verbose => true
2005-05-26 02:41:32 +04:00
# FileUtils.cp 'symlink', 'dest' # copy content, "dest" is not a symlink
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def cp ( src , dest , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :preserve , :noop , :verbose
fu_output_message " cp #{ options [ :preserve ] ? ' -p' : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
2004-08-09 01:35:11 +04:00
copy_file s , d , options [ :preserve ]
2002-03-26 05:03:04 +03:00
end
end
alias copy cp
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'cp' ] =
OPT_TABLE [ 'copy' ] = %w( noop verbose preserve )
2003-01-24 19:13:33 +03:00
#
2005-05-26 02:41:32 +04:00
# Options: preserve noop verbose dereference_root
2003-01-24 19:13:33 +03:00
#
# Copies +src+ to +dest+. If +src+ is a directory, this method copies
# all its contents recursively. If +dest+ is a directory, copies
# +src+ to +dest/src+.
#
# +src+ can be a list of files.
#
# # Installing ruby library "mylib" under the site_ruby
# FileUtils.rm_r site_ruby + '/mylib', :force
# FileUtils.cp_r 'lib/', site_ruby + '/mylib'
#
# # Examples of copying several files to target directory.
# FileUtils.cp_r %w(mail.rb field.rb debug/), site_ruby + '/tmail'
* array.c: replace rb_protect_inspect() and rb_inspecting_p() by
rb_exec_recursive() in eval.c.
* eval.c (rb_exec_recursive): new function.
* array.c (rb_ary_join): use rb_exec_recursive().
* array.c (rb_ary_inspect, rb_ary_hash): ditto.
* file.c (rb_file_join): ditto.
* hash.c (rb_hash_inspect, rb_hash_to_s, rb_hash_hash): ditto.
* io.c (rb_io_puts): ditto.
* object.c (rb_obj_inspect): ditto
* struct.c (rb_struct_inspect): ditto.
* lib/set.rb (SortedSet::setup): a hack to shut up warning.
[ruby-talk:132866]
* lib/time.rb (Time::strptime): add new function. inspired by
[ruby-talk:132815].
* lib/parsedate.rb (ParseDate::strptime): ditto.
* regparse.c: move st_*_strend() functions from st.c. fixed some
potential memory leaks.
* exception error messages updated. [ruby-core:04497]
* ext/socket/socket.c (Init_socket): add bunch of Socket
constants. Patch from Sam Roberts <sroberts@uniserve.com>.
[ruby-core:04409]
* array.c (rb_ary_s_create): no need for negative argc check.
[ruby-core:04463]
* array.c (rb_ary_unshift_m): ditto.
* lib/xmlrpc/parser.rb (XMLRPC::FaultException): make it subclass
of StandardError class, not Exception class. [ruby-core:04429]
* parse.y (fcall_gen): lvar(arg) will be evaluated as
lvar.call(arg) when lvar is a defined local variable. [new]
* object.c (rb_class_initialize): call inherited method before
calling initializing block.
* eval.c (rb_thread_start_1): initialize newly pushed frame.
* lib/open3.rb (Open3::popen3): $? should not be EXIT_FAILURE.
fixed: [ruby-core:04444]
* eval.c (is_defined): NODE_IASGN is an assignment.
* ext/readline/readline.c (Readline.readline): use rl_outstream
and rl_instream. [ruby-dev:25699]
* ext/etc/etc.c (Init_etc): sGroup needs HAVE_ST_GR_PASSWD check
[ruby-dev:25675]
* misc/ruby-mode.el: [ruby-core:04415]
* lib/rdoc/generators/html_generator.rb: [ruby-core:04412]
* lib/rdoc/generators/ri_generator.rb: ditto.
* struct.c (make_struct): fixed: [ruby-core:04402]
* ext/curses/curses.c (window_color_set): [ruby-core:04393]
* ext/socket/socket.c (Init_socket): SO_REUSEPORT added.
[ruby-talk:130092]
* object.c: [ruby-doc:818]
* parse.y (open_args): fix too verbose warnings for the space
before argument parentheses. [ruby-dev:25492]
* parse.y (parser_yylex): ditto.
* parse.y (parser_yylex): the first expression in the parentheses
should not be a command. [ruby-dev:25492]
* lib/irb/context.rb (IRB::Context::initialize): [ruby-core:04330]
* object.c (Init_Object): remove Object#type. [ruby-core:04335]
* st.c (st_foreach): report success/failure by return value.
[ruby-Bugs-1396]
* parse.y: forgot to initialize parser struct. [ruby-dev:25492]
* parse.y (parser_yylex): no tLABEL on EXPR_BEG.
[ruby-talk:127711]
* document updates - [ruby-core:04296], [ruby-core:04301],
[ruby-core:04302], [ruby-core:04307]
* dir.c (rb_push_glob): should work for NUL delimited patterns.
* dir.c (rb_glob2): should aware of offset in the pattern.
* string.c (rb_str_new4): should propagate taintedness.
* env.h: rename member names in struct FRAME; last_func -> callee,
orig_func -> this_func, last_class -> this_class.
* struct.c (rb_struct_set): use original method name, not callee
name, to retrieve member slot. [ruby-core:04268]
* time.c (time_strftime): protect from format modification from GC
finalizers.
* object.c (Init_Object): remove rb_obj_id_obsolete()
* eval.c (rb_mod_define_method): incomplete subclass check.
[ruby-dev:25464]
* gc.c (rb_data_object_alloc): klass may be NULL.
[ruby-list:40498]
* bignum.c (rb_big_rand): should return positive random number.
[ruby-dev:25401]
* bignum.c (rb_big_rand): do not use rb_big_modulo to generate
random bignums. [ruby-dev:25396]
* variable.c (rb_autoload): [ruby-dev:25373]
* eval.c (svalue_to_avalue): [ruby-dev:25366]
* string.c (rb_str_justify): [ruby-dev:25367]
* io.c (rb_f_select): [ruby-dev:25312]
* ext/socket/socket.c (sock_s_getservbyport): [ruby-talk:124072]
* struct.c (make_struct): [ruby-dev:25249]
* dir.c (dir_open_dir): new function. [ruby-dev:25242]
* io.c (rb_f_open): add type check for return value from to_open.
* lib/pstore.rb (PStore#transaction): Use the empty content when a
file is not found. [ruby-dev:24561]
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@8068 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2005-03-04 09:47:45 +03:00
# FileUtils.cp_r Dir.glob('*.rb'), '/home/aamine/lib/ruby', :noop => true, :verbose => true
2004-08-09 01:35:11 +04:00
#
# # If you want to copy all contents of a directory instead of the
# # directory itself, c.f. src/x -> dest/x, src/y -> dest/y,
# # use following code.
# FileUtils.cp_r 'src/.', 'dest' # cp_r('src', 'dest') makes src/dest,
# # but this doesn't.
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def cp_r ( src , dest , options = { } )
2005-05-26 02:41:32 +04:00
fu_check_options options , :preserve , :noop , :verbose , :dereference_root
2003-05-02 10:07:54 +04:00
fu_output_message " cp -r #{ options [ :preserve ] ? 'p' : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
copy_entry s , d , options [ :preserve ] , options [ :dereference_root ]
2002-03-26 05:03:04 +03:00
end
end
2005-05-26 02:41:32 +04:00
OPT_TABLE [ 'cp_r' ] = %w( noop verbose preserve dereference_root )
2002-03-26 05:03:04 +03:00
2004-08-09 01:35:11 +04:00
#
# Copies a file system entry +src+ to +dest+.
2005-05-26 02:41:32 +04:00
# If +src+ is a directory, this method copies its contents recursively.
# This method preserves file types, c.f. symlink, directory...
# (FIFO, device files and etc. are not supported yet)
2004-08-09 01:35:11 +04:00
#
# Both of +src+ and +dest+ must be a path name.
# +src+ must exist, +dest+ must not exist.
#
2005-03-26 18:57:16 +03:00
# If +preserve+ is true, this method preserves owner, group, permissions
# and modified time.
2004-08-09 01:35:11 +04:00
#
2005-05-26 02:41:32 +04:00
# If +dereference_root+ is true, this method dereference tree root.
#
def copy_entry ( src , dest , preserve = false , dereference_root = false )
Entry_ . new ( src , nil , dereference_root ) . traverse do | ent |
destent = Entry_ . new ( dest , ent . rel , false )
ent . copy destent . path
ent . copy_metadata destent . path if preserve
end
2002-03-26 05:03:04 +03:00
end
2003-02-21 14:57:31 +03:00
#
2004-08-09 01:35:11 +04:00
# Copies file contents of +src+ to +dest+.
# Both of +src+ and +dest+ must be a path name.
2003-02-21 14:57:31 +03:00
#
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
2003-02-21 14:57:31 +03:00
#
# Copies stream +src+ to +dest+.
2005-03-26 18:57:16 +03:00
# +src+ must respond to #read(n) and
# +dest+ must respond to #write(str).
2003-02-21 14:57:31 +03:00
#
2003-12-11 12:49:47 +03:00
def copy_stream ( src , dest )
2004-08-09 01:35:11 +04:00
fu_copy_stream0 src , dest , fu_stream_blksize ( src , dest )
end
2003-01-24 19:13:33 +03:00
#
2004-05-07 17:08:12 +04:00
# Options: force noop verbose
2003-01-24 19:13:33 +03:00
#
# Moves file(s) +src+ to +dest+. If +file+ and +dest+ exist on the different
# disk partition, the file is copied instead.
#
# FileUtils.mv 'badname.rb', 'goodname.rb'
2004-05-07 17:08:12 +04:00
# FileUtils.mv 'stuff.rb', '/notexist/lib/ruby', :force => true # no error
2003-01-24 19:13:33 +03:00
#
2003-02-06 12:56:13 +03:00
# FileUtils.mv %w(junk.txt dust.txt), '/home/aamine/.trash/'
2004-05-07 17:08:12 +04:00
# FileUtils.mv Dir.glob('test*.rb'), 'test', :noop => true, :verbose => true
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def mv ( src , dest , options = { } )
2004-05-07 17:08:12 +04:00
fu_check_options options , :force , :noop , :verbose
fu_output_message " mv #{ options [ :force ] ? ' -f' : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
2003-05-02 10:07:54 +04:00
return if options [ :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?
2005-05-26 15:33:16 +04:00
raise Errno :: EEXIST , dest
2005-05-26 02:41:32 +04:00
else
destent . remove_file if rename_cannot_overwrite_file?
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
rescue Errno :: EXDEV
2004-08-09 01:35:11 +04:00
copy_entry s , d , true
2002-03-26 05:03:04 +03:00
end
2004-12-27 09:15:32 +03:00
rescue SystemCallError
raise unless options [ :force ]
2002-03-26 05:03:04 +03:00
end
end
end
alias move mv
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'mv' ] =
OPT_TABLE [ 'move' ] = %w( noop verbose force )
2003-11-20 13:16:27 +03:00
def rename_cannot_overwrite_file? #:nodoc:
2004-12-27 09:15:32 +03:00
/ djgpp|cygwin|mswin|mingw|bccwin|wince|emx / =~ RUBY_PLATFORM
2002-10-17 07:54:25 +04:00
end
2003-11-20 13:16:27 +03:00
private :rename_cannot_overwrite_file?
2002-10-17 07:54:25 +04:00
2003-01-24 19:13:33 +03:00
#
# Options: force noop verbose
#
# Remove file(s) specified in +list+. This method cannot remove directories.
2004-08-09 01:35:11 +04:00
# All StandardErrors are ignored when the :force option is set.
2003-01-24 19:13:33 +03:00
#
# FileUtils.rm %w( junk.txt dust.txt )
2003-02-06 12:56:13 +03:00
# FileUtils.rm Dir.glob('*.so')
2003-05-02 10:07:54 +04:00
# FileUtils.rm 'NotExistFile', :force => true # never raises exception
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def rm ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :force , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " rm #{ options [ :force ] ? ' -f' : '' } #{ list . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2002-03-26 05:03:04 +03:00
2005-05-21 21:40:24 +04:00
list . each do | path |
remove_file path , options [ :force ]
2002-03-26 05:03:04 +03:00
end
end
alias remove rm
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'rm' ] =
OPT_TABLE [ 'remove' ] = %w( noop verbose force )
2003-01-24 19:13:33 +03:00
#
# Options: noop verbose
#
2005-05-26 02:41:32 +04:00
# Equivalent to
#
# #rm(list, :force => true)
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def rm_f ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :noop , :verbose
options = options . dup
options [ :force ] = true
rm list , options
2002-03-26 05:03:04 +03:00
end
alias safe_unlink rm_f
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'rm_f' ] =
OPT_TABLE [ 'safe_unlink' ] = %w( noop verbose )
2003-01-24 19:13:33 +03:00
#
2005-05-21 21:40:24 +04:00
# Options: force noop verbose secure
2003-01-24 19:13:33 +03:00
#
# remove files +list+[0] +list+[1]... If +list+[n] is a directory,
# removes its all contents recursively. This method ignores
# StandardError when :force option is set.
#
# FileUtils.rm_r Dir.glob('/tmp/*')
2003-05-02 10:07:54 +04:00
# FileUtils.rm_r '/', :force => true # :-)
2005-05-21 21:40:24 +04:00
#
2005-05-26 15:33:16 +04:00
# WARNING: This method causes local vulnerability
2005-05-26 02:41:32 +04:00
# if one of parent directories or removing directory tree are world
2005-08-13 17:14:07 +04:00
# writable (including /tmp, whose permission is 1777), and the current
# process has strong privilege such as Unix super user (root), and the
# system has symbolic link. For secure removing, read the documentation
# of #remove_entry_secure carefully, and set :secure option to true.
2005-05-26 15:33:16 +04:00
# Default is :secure=>false.
2005-05-25 14:50:14 +04:00
#
2005-05-26 02:41:32 +04:00
# NOTE: This method calls #remove_entry_secure if :secure option is set.
# See also #remove_entry_secure.
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def rm_r ( list , options = { } )
2005-05-21 21:40:24 +04:00
fu_check_options options , :force , :noop , :verbose , :secure
2005-05-26 15:33:16 +04:00
# options[:secure] = true unless options.key?(:secure)
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " rm -r #{ options [ :force ] ? 'f' : '' } #{ list . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2005-05-21 21:40:24 +04:00
list . each do | path |
2005-05-26 02:41:32 +04:00
if options [ :secure ]
remove_entry_secure path , options [ :force ]
2005-05-21 21:40:24 +04:00
else
2005-05-26 02:41:32 +04:00
remove_entry path , options [ :force ]
2005-05-21 21:40:24 +04:00
end
end
end
2005-05-26 02:41:32 +04:00
OPT_TABLE [ 'rm_r' ] = %w( noop verbose force secure )
2002-03-26 05:03:04 +03:00
2003-01-24 19:13:33 +03:00
#
2005-05-21 21:40:24 +04:00
# Options: noop verbose secure
2003-01-24 19:13:33 +03:00
#
2005-05-26 02:41:32 +04:00
# Equivalent to
#
2003-05-02 10:07:54 +04:00
# #rm_r(list, :force => true)
2005-05-25 14:50:14 +04:00
#
2005-05-26 15:33:16 +04:00
# WARNING: This method causes local vulnerability.
2005-05-25 14:50:14 +04:00
# Read the documentation of #rm_r first.
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def rm_rf ( list , options = { } )
2005-05-21 21:40:24 +04:00
fu_check_options options , :noop , :verbose , :secure
2003-05-02 10:07:54 +04:00
options = options . dup
options [ :force ] = true
rm_r list , options
2002-03-26 05:03:04 +03:00
end
2003-02-21 14:57:31 +03:00
alias rmtree rm_rf
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'rm_rf' ] =
2005-05-26 02:41:32 +04:00
OPT_TABLE [ 'rmtree' ] = %w( noop verbose secure )
2005-05-21 21:40:24 +04:00
2005-05-26 02:41:32 +04:00
#
2005-05-26 15:33:16 +04:00
# This method removes a file system entry +path+. +path+ shall be a
# regular file, a directory, or something. If +path+ is a directory,
# remove it recursively. This method is required to avoid TOCTTOU
# (time-of-check-to-time-of-use) local security vulnerability of #rm_r.
2005-08-13 17:17:23 +04:00
# #rm_r causes security hole when:
2005-05-26 15:33:16 +04:00
#
# * Parent directory is world writable (including /tmp).
# * Removing directory tree includes world writable directory.
2005-08-13 17:14:07 +04:00
# * The system has symbolic link.
2005-05-26 02:41:32 +04:00
#
2005-05-26 15:33:16 +04:00
# To avoid this security hole, this method applies special preprocess.
2005-05-26 02:41:32 +04:00
# If +path+ is a directory, this method chown(2) and chmod(2) all
2005-05-26 15:33:16 +04:00
# removing directories. This requires the current process is the
2005-08-13 17:06:04 +04:00
# owner of the removing whole directory tree, or is the super user (root).
2005-05-26 02:41:32 +04:00
#
# WARNING: You must ensure that *ALL* parent directories are not
# world writable. Otherwise this method does not work.
2005-08-13 17:06:04 +04:00
# Only exception is temporary directory like /tmp and /var/tmp,
# whose permission is 1777.
2005-05-26 02:41:32 +04:00
#
2005-08-13 17:14:07 +04:00
# WARNING: Only the owner of the removing directory tree, or Unix super
# user (root) should invoke this method. Otherwise this method does not
# work.
2005-05-26 02:41:32 +04:00
#
# For details of this security vulnerability, see Perl's case:
#
# http://www.cve.mitre.org/cgi-bin/cvename.cgi?name=CAN-2005-0448
# http://www.cve.mitre.org/cgi-bin/cvename.cgi?name=CAN-2004-0452
#
# For fileutils.rb, this vulnerability is reported in [ruby-dev:26100].
#
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
2005-09-19 00:33:01 +04:00
# freeze tree root
2005-08-13 17:06:04 +04:00
euid = Process . euid
2005-09-19 00:33:01 +04:00
File . open ( fullpath + '/.' ) { | f |
unless fu_stat_identical_entry? ( st , f . stat )
# symlink (TOC-to-TOU attack?)
2005-08-13 17:06:04 +04:00
File . unlink fullpath
return
end
2005-09-19 00:33:01 +04:00
f . chown euid , - 1
f . chmod 0700
}
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
2005-09-19 00:33:01 +04:00
def fu_have_symlink?
File . symlink nil , nil
rescue NotImplementedError
return false
rescue
return true
end
2005-08-13 17:06:04 +04:00
def fu_stat_identical_entry? ( a , b )
a . dev == b . dev and a . ino == b . ino
2005-05-26 02:41:32 +04:00
end
2005-09-19 00:33:01 +04:00
private :fu_stat_identical_entry?
2005-05-26 02:41:32 +04:00
#
# This method removes a file system entry +path+.
# +path+ might be a regular file, a directory, or something.
# If +path+ is a directory, remove it recursively.
#
# See also #remove_entry_secure.
#
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
#
# Removes a file +path+.
# This method ignores StandardError if +force+ is true.
#
def remove_file ( path , force = false )
Entry_ . new ( path ) . remove_file
rescue
raise unless force
end
#
# Removes a directory +dir+ and its contents recursively.
# This method ignores StandardError if +force+ is true.
#
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
2003-01-24 19:13:33 +03:00
#
# Returns true if the contents of a file A and a file B are identical.
#
2003-02-21 14:57:31 +03:00
# FileUtils.compare_file('somefile', 'somefile') #=> true
# FileUtils.compare_file('/bin/cp', '/bin/mv') #=> maybe false
2003-01-24 19:13:33 +03:00
#
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
2002-03-26 05:03:04 +03:00
2003-02-21 14:57:31 +03:00
alias identical? compare_file
alias cmp compare_file
2002-03-26 05:03:04 +03:00
2003-02-21 14:57:31 +03:00
#
# Returns true if the contents of a stream +a+ and +b+ are identical.
#
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 )
sa = sb = nil
while sa == sb
sa = a . read ( bsize )
sb = b . read ( bsize )
unless sa and sb
if sa . nil? and sb . nil?
return true
2002-03-26 05:03:04 +03:00
end
end
2003-02-21 14:57:31 +03:00
end
2002-03-26 05:03:04 +03:00
false
end
2003-01-24 19:13:33 +03:00
#
2003-05-02 10:07:54 +04:00
# Options: mode noop verbose
2003-01-24 19:13:33 +03:00
#
# If +src+ is not same as +dest+, copies it and changes the permission
2003-02-06 12:56:13 +03:00
# mode to +mode+. If +dest+ is a directory, destination is +dest+/+src+.
2003-01-24 19:13:33 +03:00
#
2003-05-02 10:07:54 +04:00
# FileUtils.install 'ruby', '/usr/local/bin/ruby', :mode => 0755, :verbose => true
# FileUtils.install 'lib.rb', '/usr/local/lib/ruby/site_ruby', :verbose => true
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def install ( src , dest , options = { } )
2003-07-29 14:24:28 +04:00
fu_check_options options , :mode , :preserve , :noop , :verbose
fu_output_message " install -c #{ options [ :preserve ] && ' -p' } #{ options [ :mode ] ? ( ' -m 0%o' % options [ :mode ] ) : '' } #{ [ src , dest ] . flatten . join ' ' } " if options [ :verbose ]
2003-05-02 10:07:54 +04:00
return if options [ :noop ]
2005-05-26 02:41:32 +04:00
fu_each_src_dest ( src , dest ) do | s , d |
unless File . exist? ( d ) and compare_file ( s , d )
2003-12-12 13:33:24 +03:00
remove_file d , true
st = File . stat ( s ) if options [ :preserve ]
copy_file s , d
File . utime st . atime , st . mtime , d if options [ :preserve ]
File . chmod options [ :mode ] , d if options [ :mode ]
2002-03-26 05:03:04 +03:00
end
end
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'install' ] = %w( noop verbose preserve mode )
2003-01-24 19:13:33 +03:00
#
# Options: noop verbose
#
# Changes permission bits on the named files (in +list+) to the bit pattern
# represented by +mode+.
#
# FileUtils.chmod 0755, 'somecommand'
2003-05-02 10:07:54 +04:00
# FileUtils.chmod 0644, %w(my.rb your.rb his.rb her.rb)
# FileUtils.chmod 0755, '/usr/bin/ruby', :verbose => true
2003-01-24 19:13:33 +03:00
#
2003-12-11 12:49:47 +03:00
def chmod ( mode , list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message sprintf ( 'chmod %o %s' , mode , list . join ( ' ' ) ) if options [ :verbose ]
return if options [ :noop ]
2005-05-26 02:41:32 +04:00
list . each do | path |
Entry_ . new ( path ) . chmod mode
end
2002-03-26 05:03:04 +03:00
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'chmod' ] = %w( noop verbose )
#
# Options: noop verbose force
#
# Changes permission bits on the named files (in +list+)
# to the bit pattern represented by +mode+.
#
# FileUtils.chmod_R 0700, "/tmp/app.#{$$}"
#
def chmod_R ( mode , list , options = { } )
fu_check_options options , :noop , :verbose , :force
list = fu_list ( list )
fu_output_message sprintf ( 'chmod -R%s %o %s' ,
( options [ :force ] ? 'f' : '' ) ,
mode , list . join ( ' ' ) ) if options [ :verbose ]
return if options [ :noop ]
2005-05-26 02:41:32 +04:00
list . each do | root |
Entry_ . new ( root ) . traverse do | ent |
begin
ent . chmod mode
rescue
raise unless options [ :force ]
2005-05-21 21:40:24 +04:00
end
end
end
end
OPT_TABLE [ 'chmod_R' ] = %w( noop verbose )
2004-12-04 14:46:24 +03:00
#
# Options: noop verbose
#
# Changes owner and group on the named files (in +list+)
2004-12-04 15:29:55 +03:00
# to the user +user+ and the group +group+. +user+ and +group+
2004-12-04 14:46:24 +03:00
# may be an ID (Integer/String) or a name (String).
# If +user+ or +group+ is nil, this method does not change
# the attribute.
#
# FileUtils.chown 'root', 'staff', '/usr/local/bin/ruby'
# FileUtils.chown nil, 'bin', Dir.glob('/usr/bin/*'), :verbose => true
#
def chown ( user , group , list , options = { } )
fu_check_options options , :noop , :verbose
list = fu_list ( list )
fu_output_message sprintf ( 'chown %s%s' ,
[ user , group ] . compact . join ( ':' ) + ' ' ,
list . join ( ' ' ) ) if options [ :verbose ]
return if options [ :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
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'chown' ] = %w( noop verbose )
2004-12-04 14:46:24 +03:00
#
2005-05-21 21:40:24 +04:00
# Options: noop verbose force
2004-12-04 14:46:24 +03:00
#
# Changes owner and group on the named files (in +list+)
2004-12-04 15:29:55 +03:00
# to the user +user+ and the group +group+ recursively.
# +user+ and +group+ may be an ID (Integer/String) or
# a name (String). If +user+ or +group+ is nil, this
# method does not change the attribute.
2004-12-04 14:46:24 +03:00
#
2004-12-04 15:29:55 +03:00
# FileUtils.chown_R 'www', 'www', '/var/www/htdocs'
# FileUtils.chown_R 'cvs', 'cvs', '/var/cvs', :verbose => true
2004-12-04 14:46:24 +03:00
#
2005-05-21 21:40:24 +04:00
def chown_R ( user , group , list , options = { } )
fu_check_options options , :noop , :verbose , :force
2004-12-04 14:46:24 +03:00
list = fu_list ( list )
2005-05-21 21:40:24 +04:00
fu_output_message sprintf ( 'chown -R%s %s%s' ,
( options [ :force ] ? 'f' : '' ) ,
2004-12-04 14:46:24 +03:00
[ user , group ] . compact . join ( ':' ) + ' ' ,
list . join ( ' ' ) ) if options [ :verbose ]
return if options [ :noop ]
uid = fu_get_uid ( user )
gid = fu_get_gid ( group )
return unless uid or gid
2005-05-26 02:41:32 +04:00
list . each do | root |
Entry_ . new ( root ) . traverse do | ent |
begin
ent . chown uid , gid
rescue
raise unless options [ :force ]
2005-05-21 21:40:24 +04:00
end
2004-12-04 14:46:24 +03:00
end
end
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'chown_R' ] = %w( noop verbose )
2004-12-04 14:46:24 +03:00
begin
require 'etc'
2005-05-26 02:41:32 +04:00
def fu_get_uid ( user ) #:nodoc:
2004-12-04 14:46:24 +03:00
return nil unless user
user = user . to_s
if / \ A \ d+ \ z / =~ user
then user . to_i
else Etc . getpwnam ( user ) . uid
end
end
private :fu_get_uid
2005-05-26 02:41:32 +04:00
def fu_get_gid ( group ) #:nodoc:
2004-12-04 14:46:24 +03:00
return nil unless group
if / \ A \ d+ \ z / =~ group
then group . to_i
else Etc . getgrnam ( group ) . gid
end
end
private :fu_get_gid
rescue LoadError
# need Win32 support???
2005-05-26 02:41:32 +04:00
def fu_get_uid ( user ) #:nodoc:
2004-12-04 14:46:24 +03:00
user # FIXME
end
2005-05-26 02:41:32 +04:00
def fu_get_gid ( group ) #:nodoc:
2004-12-04 14:46:24 +03:00
group # FIXME
end
end
2003-01-24 19:13:33 +03:00
#
# Options: noop verbose
#
# Updates modification time (mtime) and access time (atime) of file(s) in
# +list+. Files are created if they don't exist.
#
# FileUtils.touch 'timestamp'
# FileUtils.touch Dir.glob('*.c'); system 'make'
#
2003-12-11 12:49:47 +03:00
def touch ( list , options = { } )
2003-05-02 10:07:54 +04:00
fu_check_options options , :noop , :verbose
2002-03-26 05:03:04 +03:00
list = fu_list ( list )
2003-05-02 10:07:54 +04:00
fu_output_message " touch #{ list . join ' ' } " if options [ :verbose ]
return if options [ :noop ]
2002-03-26 05:03:04 +03:00
t = Time . now
2005-05-21 21:40:24 +04:00
list . each do | path |
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
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
}
2002-03-26 05:03:04 +03:00
end
end
end
2005-05-21 21:40:24 +04:00
OPT_TABLE [ 'touch' ] = %w( noop verbose )
2002-03-26 05:03:04 +03:00
private
2005-05-26 02:41:32 +04:00
module StreamUtils_
private
def fu_windows?
/ mswin|mingw|bccwin|wince|emx / =~ RUBY_PLATFORM
end
def fu_copy_stream0 ( src , dest , blksize ) #:nodoc:
# FIXME: readpartial?
while s = src . read ( blksize )
dest . write s
2005-05-21 21:40:24 +04:00
end
end
2005-05-26 02:41:32 +04:00
def fu_stream_blksize ( * streams )
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
2005-05-26 02:41:32 +04:00
def fu_blksize ( st )
s = st . blksize
return nil unless s
return nil if s == 0
s
end
2005-05-21 21:40:24 +04:00
2005-05-26 02:41:32 +04:00
def fu_default_blksize
1024
2002-03-26 05:03:04 +03:00
end
end
2005-05-26 02:41:32 +04:00
include StreamUtils_
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
@path
else
join ( @prefix , @rel )
end
end
def prefix
@prefix || @path
end
def rel
@rel
end
def dereference?
@deref
end
def exist?
lstat! ? true : false
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
Dir . entries ( path ( ) ) \
. reject { | n | n == '.' or n == '..' } \
. map { | n | Entry_ . new ( prefix ( ) , join ( rel ( ) , n . untaint ) ) }
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
end
def chown ( uid , gid )
if symlink?
File . lchown uid , gid , path ( ) if have_lchown?
else
File . chown uid , gid , path ( )
end
end
def copy ( dest )
case
when file?
copy_file dest
when directory?
begin
Dir . mkdir dest
rescue
raise unless File . directory? ( dest )
end
when symlink?
File . symlink File . readlink ( path ( ) ) , dest
when chardev?
raise " cannot handle device file " unless File . respond_to? ( :mknod )
mknod dest , ?c , 0666 , lstat ( ) . rdev
when blockdev?
raise " cannot handle device file " unless File . respond_to? ( :mknod )
mknod dest , ?b , 0666 , lstat ( ) . rdev
when socket?
raise " cannot handle socket " unless File . respond_to? ( :mknod )
mknod dest , nil , lstat ( ) . mode , 0
when pipe?
raise " cannot handle FIFO " unless File . respond_to? ( :mkfifo )
mkfifo dest , 0666
when door?
raise " cannot handle door: #{ path ( ) } "
else
raise " unknown file type: #{ path ( ) } "
end
end
def copy_file ( dest )
st = stat ( )
File . open ( path ( ) , 'rb' ) { | r |
File . open ( dest , 'wb' , st . mode ) { | w |
fu_copy_stream0 r , w , ( fu_blksize ( st ) || fu_default_blksize ( ) )
}
}
end
def copy_metadata ( path )
st = lstat ( )
File . utime st . atime , st . mtime , path
begin
File . chown st . uid , st . gid , path
rescue Errno :: EPERM
# clear setuid/setgid
File . chmod st . mode & 01777 , path
else
File . chmod st . mode , path
end
end
def remove
if directory?
remove_dir1
else
remove_file
end
end
def remove_dir1
platform_support {
Dir . rmdir path ( ) . sub ( %r< / \ z > , '' )
}
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?
entries ( ) . each do | ent |
ent . postorder_traverse do | e |
yield e
end
end
end
yield self
end
private
$fileutils_rb_have_lchmod = nil
def have_lchmod?
# This is not MT-safe, but it does not matter.
if $fileutils_rb_have_lchmod == nil
$fileutils_rb_have_lchmod = check_have_lchmod?
end
$fileutils_rb_have_lchmod
end
def check_have_lchmod?
return false unless File . respond_to? ( :lchmod )
File . lchmod 0
return true
rescue NotImplementedError
return false
end
$fileutils_rb_have_lchown = nil
def have_lchown?
# This is not MT-safe, but it does not matter.
if $fileutils_rb_have_lchown == nil
$fileutils_rb_have_lchown = check_have_lchown?
end
$fileutils_rb_have_lchown
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 == '.'
File . join ( dir , base )
end
end # class Entry_
2003-12-11 12:49:47 +03:00
def fu_list ( arg )
2004-05-07 16:04:19 +04:00
[ arg ] . flatten . map { | path | File . path ( path ) }
2002-03-26 05:03:04 +03:00
end
2003-12-11 12:49:47 +03:00
def fu_each_src_dest ( src , dest )
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 )
yield s , d
end
end
2003-12-11 12:49:47 +03:00
def fu_each_src_dest0 ( src , dest )
2003-12-12 13:33:24 +03:00
if src . is_a? ( Array )
src . each do | s |
2004-04-07 06:51:05 +04:00
s = File . path ( s )
yield s , File . join ( dest , File . basename ( s ) )
2002-03-26 05:03:04 +03:00
end
else
2004-04-07 06:51:05 +04:00
src = File . path ( src )
2003-12-12 13:33:24 +03:00
if 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
2003-12-11 12:49:47 +03:00
def fu_same? ( a , b )
2005-05-26 02:41:32 +04:00
if fu_have_st_ino?
2003-11-19 08:33:07 +03:00
st1 = File . stat ( a )
st2 = File . stat ( b )
st1 . dev == st2 . dev and st1 . ino == st2 . ino
2003-11-18 16:24:48 +03:00
else
File . expand_path ( a ) == File . expand_path ( b )
end
2003-11-18 13:05:21 +03:00
rescue Errno :: ENOENT
return false
2003-11-18 08:09:20 +03:00
end
2005-05-26 02:41:32 +04:00
def fu_have_st_ino?
not fu_windows?
2003-11-18 16:24:48 +03:00
end
2005-05-26 02:41:32 +04:00
def fu_check_options ( options , * optdecl )
h = options . dup
optdecl . each do | name |
h . delete name
2002-10-17 07:54:25 +04:00
end
2005-05-26 02:41:32 +04:00
raise ArgumentError , " no such option: #{ h . keys . join ( ' ' ) } " unless h . empty?
2004-08-09 01:35:11 +04:00
end
2005-05-26 02:41:32 +04:00
def fu_update_option ( args , new )
if args . last . is_a? ( Hash )
args [ - 1 ] = args . last . dup . update ( new )
else
args . push new
end
args
2002-10-17 07:54:25 +04:00
end
2002-03-26 05:03:04 +03:00
@fileutils_output = $stderr
2003-02-06 12:56:13 +03:00
@fileutils_label = ''
2002-03-26 05:03:04 +03:00
2003-12-11 12:49:47 +03:00
def fu_output_message ( msg )
2002-03-26 05:03:04 +03:00
@fileutils_output || = $stderr
2003-02-06 12:56:13 +03:00
@fileutils_label || = ''
2002-03-26 05:03:04 +03:00
@fileutils_output . puts @fileutils_label + msg
end
2005-05-21 21:40:24 +04:00
# All Methods are public instance method and are public class method.
2002-03-26 05:03:04 +03:00
extend self
2004-05-07 17:08:12 +04:00
#
# Returns an Array of method names which have any options.
#
# p FileUtils.commands #=> ["chmod", "cp", "cp_r", "install", ...]
#
def FileUtils . commands
OPT_TABLE . keys
end
#
# Returns an Array of option names.
#
# p FileUtils.options #=> ["noop", "force", "verbose", "preserve", "mode"]
#
def FileUtils . options
OPT_TABLE . values . flatten . uniq
end
#
# Returns true if the method +mid+ have an option +opt+.
#
# p FileUtils.have_option?(:cp, :noop) #=> true
# p FileUtils.have_option?(:rm, :force) #=> true
# p FileUtils.have_option?(:rm, :perserve) #=> false
#
def FileUtils . 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 } "
2004-05-07 17:08:12 +04:00
li . include? ( opt . to_s )
end
#
# Returns an Array of option names of the method +mid+.
#
# p FileUtils.options(:rm) #=> ["noop", "verbose", "force"]
#
def FileUtils . options_of ( mid )
OPT_TABLE [ mid . to_s ]
end
#
# Returns an Array of method names which have the option +opt+.
#
# p FileUtils.collect_methods(:preserve) #=> ["cp", "cp_r", "copy", "install"]
#
def FileUtils . collect_methods ( opt )
2005-05-26 02:41:32 +04:00
OPT_TABLE . keys . select { | m | OPT_TABLE [ m ] . include? ( opt . to_s ) }
2004-05-07 17:08:12 +04:00
end
2003-01-24 19:13:33 +03:00
#
2003-02-06 12:56:13 +03:00
# This module has all methods of FileUtils module, but it outputs messages
2004-05-07 17:08:12 +04:00
# before acting. This equates to passing the <tt>:verbose</tt> flag to
# methods in FileUtils.
2003-01-24 19:13:33 +03:00
#
2002-03-26 05:03:04 +03:00
module Verbose
include FileUtils
@fileutils_output = $stderr
2003-02-06 12:56:13 +03:00
@fileutils_label = ''
2005-05-26 02:41:32 +04:00
:: FileUtils . collect_methods ( 'verbose' ) . each do | name |
2003-02-06 12:56:13 +03:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
2003-12-11 12:49:47 +03:00
def #{name}(*args)
2004-05-07 17:08:12 +04:00
super ( * fu_update_option ( args , :verbose = > true ) )
2003-02-06 12:56:13 +03:00
end
EOS
2002-03-26 05:03:04 +03:00
end
extend self
end
2003-01-24 19:13:33 +03:00
#
2003-02-06 12:56:13 +03:00
# This module has all methods of FileUtils module, but never changes
2004-05-07 17:08:12 +04:00
# files/directories. This equates to passing the <tt>:noop</tt> flag
# to methods in FileUtils.
2003-01-24 19:13:33 +03:00
#
2002-03-26 05:03:04 +03:00
module NoWrite
include FileUtils
@fileutils_output = $stderr
2003-02-06 12:56:13 +03:00
@fileutils_label = ''
2005-05-26 02:41:32 +04:00
:: FileUtils . collect_methods ( 'noop' ) . each do | name |
2003-02-06 12:56:13 +03:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
2003-12-11 12:49:47 +03:00
def #{name}(*args)
2003-05-02 10:07:54 +04:00
super ( * fu_update_option ( args , :noop = > true ) )
2003-02-06 12:56:13 +03:00
end
EOS
2002-03-26 05:03:04 +03:00
end
extend self
end
2004-02-16 19:23:07 +03:00
#
# This module has all methods of FileUtils module, but never changes
# files/directories, with printing message before acting.
2004-05-07 17:08:12 +04:00
# This equates to passing the <tt>:noop</tt> and <tt>:verbose</tt> flag
2004-02-16 19:23:07 +03:00
# to methods in FileUtils.
#
module DryRun
include FileUtils
@fileutils_output = $stderr
@fileutils_label = ''
2005-05-26 02:41:32 +04:00
:: FileUtils . collect_methods ( 'noop' ) . each do | name |
2004-02-16 19:23:07 +03:00
module_eval ( <<-EOS, __FILE__, __LINE__ + 1)
def #{name}(*args)
super ( * fu_update_option ( args , :noop = > true , :verbose = > true ) )
end
EOS
end
extend self
end
end