vergissberlin 0.0.5 → 0.0.7

Sign up to get free protection for your applications and to get access to all the features.
Files changed (127) hide show
  1. checksums.yaml +4 -4
  2. data/Rakefile +10 -1
  3. data/bin/vergissberlin +1 -1
  4. data/lib/vergissberlin/version.rb +1 -1
  5. data/test/test_test.rb +7 -0
  6. data/test/utils.rb +37 -0
  7. data/vendor/bundle/bin/rake +23 -0
  8. data/vendor/bundle/bin/vergissberlin +23 -0
  9. data/vendor/bundle/build_info/capistrano_colors-0.5.5.info +1 -0
  10. data/vendor/bundle/build_info/rake-10.0.4.info +1 -0
  11. data/vendor/bundle/gems/capistrano_colors-0.5.5/Manifest +7 -0
  12. data/vendor/bundle/gems/capistrano_colors-0.5.5/README.rdoc +130 -0
  13. data/vendor/bundle/gems/capistrano_colors-0.5.5/Rakefile +1 -0
  14. data/vendor/bundle/gems/capistrano_colors-0.5.5/capistrano_colors.gemspec +33 -0
  15. data/vendor/bundle/gems/capistrano_colors-0.5.5/lib/capistrano_colors/configuration.rb +70 -0
  16. data/vendor/bundle/gems/capistrano_colors-0.5.5/lib/capistrano_colors/logger.rb +72 -0
  17. data/vendor/bundle/gems/capistrano_colors-0.5.5/lib/capistrano_colors.rb +18 -0
  18. data/vendor/bundle/gems/rake-10.0.4/CHANGES +530 -0
  19. data/vendor/bundle/gems/rake-10.0.4/MIT-LICENSE +21 -0
  20. data/vendor/bundle/gems/rake-10.0.4/README.rdoc +187 -0
  21. data/vendor/bundle/gems/rake-10.0.4/Rakefile +375 -0
  22. data/vendor/bundle/gems/rake-10.0.4/TODO +21 -0
  23. data/vendor/bundle/gems/rake-10.0.4/bin/rake +33 -0
  24. data/vendor/bundle/gems/rake-10.0.4/install.rb +90 -0
  25. data/vendor/bundle/gems/rake-10.0.4/lib/rake/alt_system.rb +109 -0
  26. data/vendor/bundle/gems/rake-10.0.4/lib/rake/application.rb +681 -0
  27. data/vendor/bundle/gems/rake-10.0.4/lib/rake/backtrace.rb +19 -0
  28. data/vendor/bundle/gems/rake-10.0.4/lib/rake/clean.rb +32 -0
  29. data/vendor/bundle/gems/rake-10.0.4/lib/rake/cloneable.rb +16 -0
  30. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/compositepublisher.rb +21 -0
  31. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/ftptools.rb +151 -0
  32. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/publisher.rb +73 -0
  33. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/rubyforgepublisher.rb +16 -0
  34. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/sshpublisher.rb +50 -0
  35. data/vendor/bundle/gems/rake-10.0.4/lib/rake/contrib/sys.rb +1 -0
  36. data/vendor/bundle/gems/rake-10.0.4/lib/rake/default_loader.rb +10 -0
  37. data/vendor/bundle/gems/rake-10.0.4/lib/rake/dsl_definition.rb +156 -0
  38. data/vendor/bundle/gems/rake-10.0.4/lib/rake/early_time.rb +18 -0
  39. data/vendor/bundle/gems/rake-10.0.4/lib/rake/ext/core.rb +27 -0
  40. data/vendor/bundle/gems/rake-10.0.4/lib/rake/ext/module.rb +0 -0
  41. data/vendor/bundle/gems/rake-10.0.4/lib/rake/ext/string.rb +168 -0
  42. data/vendor/bundle/gems/rake-10.0.4/lib/rake/ext/time.rb +15 -0
  43. data/vendor/bundle/gems/rake-10.0.4/lib/rake/file_creation_task.rb +24 -0
  44. data/vendor/bundle/gems/rake-10.0.4/lib/rake/file_list.rb +410 -0
  45. data/vendor/bundle/gems/rake-10.0.4/lib/rake/file_task.rb +47 -0
  46. data/vendor/bundle/gems/rake-10.0.4/lib/rake/file_utils.rb +114 -0
  47. data/vendor/bundle/gems/rake-10.0.4/lib/rake/file_utils_ext.rb +143 -0
  48. data/vendor/bundle/gems/rake-10.0.4/lib/rake/gempackagetask.rb +1 -0
  49. data/vendor/bundle/gems/rake-10.0.4/lib/rake/invocation_chain.rb +51 -0
  50. data/vendor/bundle/gems/rake-10.0.4/lib/rake/invocation_exception_mixin.rb +16 -0
  51. data/vendor/bundle/gems/rake-10.0.4/lib/rake/loaders/makefile.rb +40 -0
  52. data/vendor/bundle/gems/rake-10.0.4/lib/rake/multi_task.rb +13 -0
  53. data/vendor/bundle/gems/rake-10.0.4/lib/rake/name_space.rb +25 -0
  54. data/vendor/bundle/gems/rake-10.0.4/lib/rake/packagetask.rb +185 -0
  55. data/vendor/bundle/gems/rake-10.0.4/lib/rake/pathmap.rb +1 -0
  56. data/vendor/bundle/gems/rake-10.0.4/lib/rake/phony.rb +15 -0
  57. data/vendor/bundle/gems/rake-10.0.4/lib/rake/private_reader.rb +20 -0
  58. data/vendor/bundle/gems/rake-10.0.4/lib/rake/promise.rb +99 -0
  59. data/vendor/bundle/gems/rake-10.0.4/lib/rake/pseudo_status.rb +24 -0
  60. data/vendor/bundle/gems/rake-10.0.4/lib/rake/rake_module.rb +37 -0
  61. data/vendor/bundle/gems/rake-10.0.4/lib/rake/rake_test_loader.rb +22 -0
  62. data/vendor/bundle/gems/rake-10.0.4/lib/rake/rdoctask.rb +1 -0
  63. data/vendor/bundle/gems/rake-10.0.4/lib/rake/ruby182_test_unit_fix.rb +25 -0
  64. data/vendor/bundle/gems/rake-10.0.4/lib/rake/rule_recursion_overflow_error.rb +20 -0
  65. data/vendor/bundle/gems/rake-10.0.4/lib/rake/runtest.rb +22 -0
  66. data/vendor/bundle/gems/rake-10.0.4/lib/rake/task.rb +368 -0
  67. data/vendor/bundle/gems/rake-10.0.4/lib/rake/task_argument_error.rb +7 -0
  68. data/vendor/bundle/gems/rake-10.0.4/lib/rake/task_arguments.rb +78 -0
  69. data/vendor/bundle/gems/rake-10.0.4/lib/rake/task_manager.rb +296 -0
  70. data/vendor/bundle/gems/rake-10.0.4/lib/rake/tasklib.rb +22 -0
  71. data/vendor/bundle/gems/rake-10.0.4/lib/rake/testtask.rb +198 -0
  72. data/vendor/bundle/gems/rake-10.0.4/lib/rake/thread_history_display.rb +48 -0
  73. data/vendor/bundle/gems/rake-10.0.4/lib/rake/thread_pool.rb +155 -0
  74. data/vendor/bundle/gems/rake-10.0.4/lib/rake/trace_output.rb +22 -0
  75. data/vendor/bundle/gems/rake-10.0.4/lib/rake/version.rb +13 -0
  76. data/vendor/bundle/gems/rake-10.0.4/lib/rake/win32.rb +55 -0
  77. data/vendor/bundle/gems/rake-10.0.4/lib/rake.rb +71 -0
  78. data/vendor/bundle/gems/rake-10.0.4/test/file_creation.rb +34 -0
  79. data/vendor/bundle/gems/rake-10.0.4/test/helper.rb +562 -0
  80. data/vendor/bundle/gems/rake-10.0.4/test/test_private_reader.rb +42 -0
  81. data/vendor/bundle/gems/rake-10.0.4/test/test_rake.rb +40 -0
  82. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_application.rb +515 -0
  83. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_application_options.rb +455 -0
  84. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_backtrace.rb +89 -0
  85. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_clean.rb +14 -0
  86. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_definitions.rb +80 -0
  87. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_directory_task.rb +57 -0
  88. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_dsl.rb +40 -0
  89. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_early_time.rb +31 -0
  90. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_extension.rb +59 -0
  91. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_file_creation_task.rb +56 -0
  92. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_file_list.rb +628 -0
  93. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_file_list_path_map.rb +8 -0
  94. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_file_task.rb +122 -0
  95. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_file_utils.rb +305 -0
  96. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_ftp_file.rb +59 -0
  97. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_functional.rb +496 -0
  98. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_invocation_chain.rb +52 -0
  99. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_makefile_loader.rb +44 -0
  100. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_multi_task.rb +59 -0
  101. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_name_space.rb +43 -0
  102. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_package_task.rb +79 -0
  103. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_path_map.rb +157 -0
  104. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_path_map_explode.rb +34 -0
  105. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_path_map_partial.rb +18 -0
  106. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_pseudo_status.rb +21 -0
  107. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_rake_test_loader.rb +21 -0
  108. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_reduce_compat.rb +30 -0
  109. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_require.rb +40 -0
  110. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_rules.rb +327 -0
  111. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task.rb +340 -0
  112. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_argument_parsing.rb +103 -0
  113. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_arguments.rb +88 -0
  114. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_lib.rb +9 -0
  115. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_manager.rb +157 -0
  116. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_manager_argument_resolution.rb +19 -0
  117. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_task_with_arguments.rb +171 -0
  118. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_test_task.rb +120 -0
  119. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_thread_pool.rb +122 -0
  120. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_top_level_functions.rb +71 -0
  121. data/vendor/bundle/gems/rake-10.0.4/test/test_rake_win32.rb +72 -0
  122. data/vendor/bundle/gems/rake-10.0.4/test/test_thread_history_display.rb +91 -0
  123. data/vendor/bundle/gems/rake-10.0.4/test/test_trace_output.rb +50 -0
  124. data/vendor/bundle/specifications/capistrano_colors-0.5.5.gemspec +32 -0
  125. data/vendor/bundle/specifications/rake-10.0.4.gemspec +35 -0
  126. data/vergissberlin.gemspec +4 -2
  127. metadata +125 -2
@@ -0,0 +1,18 @@
1
+ module Rake
2
+
3
+ # EarlyTime is a fake timestamp that occurs _before_ any other time value.
4
+ class EarlyTime
5
+ include Comparable
6
+ include Singleton
7
+
8
+ def <=>(other)
9
+ -1
10
+ end
11
+
12
+ def to_s
13
+ "<EARLY TIME>"
14
+ end
15
+ end
16
+
17
+ EARLY = EarlyTime.instance
18
+ end
@@ -0,0 +1,27 @@
1
+ ######################################################################
2
+ # Core extension library
3
+ #
4
+ class Module
5
+ # Check for an existing method in the current class before extending. IF
6
+ # the method already exists, then a warning is printed and the extension is
7
+ # not added. Otherwise the block is yielded and any definitions in the
8
+ # block will take effect.
9
+ #
10
+ # Usage:
11
+ #
12
+ # class String
13
+ # rake_extension("xyz") do
14
+ # def xyz
15
+ # ...
16
+ # end
17
+ # end
18
+ # end
19
+ #
20
+ def rake_extension(method)
21
+ if method_defined?(method)
22
+ $stderr.puts "WARNING: Possible conflict with Rake extension: #{self}##{method} already exists"
23
+ else
24
+ yield
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,168 @@
1
+ require 'rake/ext/core'
2
+
3
+ ######################################################################
4
+ # Rake extension methods for String.
5
+ #
6
+ class String
7
+
8
+ rake_extension("ext") do
9
+ # Replace the file extension with +newext+. If there is no extension on
10
+ # the string, append the new extension to the end. If the new extension
11
+ # is not given, or is the empty string, remove any existing extension.
12
+ #
13
+ # +ext+ is a user added method for the String class.
14
+ def ext(newext='')
15
+ return self.dup if ['.', '..'].include? self
16
+ if newext != ''
17
+ newext = (newext =~ /^\./) ? newext : ("." + newext)
18
+ end
19
+ self.chomp(File.extname(self)) << newext
20
+ end
21
+ end
22
+
23
+ rake_extension("pathmap") do
24
+ # Explode a path into individual components. Used by +pathmap+.
25
+ def pathmap_explode
26
+ head, tail = File.split(self)
27
+ return [self] if head == self
28
+ return [tail] if head == '.' || tail == '/'
29
+ return [head, tail] if head == '/'
30
+ return head.pathmap_explode + [tail]
31
+ end
32
+ protected :pathmap_explode
33
+
34
+ # Extract a partial path from the path. Include +n+ directories from the
35
+ # front end (left hand side) if +n+ is positive. Include |+n+|
36
+ # directories from the back end (right hand side) if +n+ is negative.
37
+ def pathmap_partial(n)
38
+ dirs = File.dirname(self).pathmap_explode
39
+ partial_dirs =
40
+ if n > 0
41
+ dirs[0...n]
42
+ elsif n < 0
43
+ dirs.reverse[0...-n].reverse
44
+ else
45
+ "."
46
+ end
47
+ File.join(partial_dirs)
48
+ end
49
+ protected :pathmap_partial
50
+
51
+ # Preform the pathmap replacement operations on the given path. The
52
+ # patterns take the form 'pat1,rep1;pat2,rep2...'.
53
+ def pathmap_replace(patterns, &block)
54
+ result = self
55
+ patterns.split(';').each do |pair|
56
+ pattern, replacement = pair.split(',')
57
+ pattern = Regexp.new(pattern)
58
+ if replacement == '*' && block_given?
59
+ result = result.sub(pattern, &block)
60
+ elsif replacement
61
+ result = result.sub(pattern, replacement)
62
+ else
63
+ result = result.sub(pattern, '')
64
+ end
65
+ end
66
+ result
67
+ end
68
+ protected :pathmap_replace
69
+
70
+ # Map the path according to the given specification. The specification
71
+ # controls the details of the mapping. The following special patterns are
72
+ # recognized:
73
+ #
74
+ # * <b>%p</b> -- The complete path.
75
+ # * <b>%f</b> -- The base file name of the path, with its file extension,
76
+ # but without any directories.
77
+ # * <b>%n</b> -- The file name of the path without its file extension.
78
+ # * <b>%d</b> -- The directory list of the path.
79
+ # * <b>%x</b> -- The file extension of the path. An empty string if there
80
+ # is no extension.
81
+ # * <b>%X</b> -- Everything *but* the file extension.
82
+ # * <b>%s</b> -- The alternate file separator if defined, otherwise use
83
+ # the standard file separator.
84
+ # * <b>%%</b> -- A percent sign.
85
+ #
86
+ # The %d specifier can also have a numeric prefix (e.g. '%2d'). If the
87
+ # number is positive, only return (up to) +n+ directories in the path,
88
+ # starting from the left hand side. If +n+ is negative, return (up to)
89
+ # |+n+| directories from the right hand side of the path.
90
+ #
91
+ # Examples:
92
+ #
93
+ # 'a/b/c/d/file.txt'.pathmap("%2d") => 'a/b'
94
+ # 'a/b/c/d/file.txt'.pathmap("%-2d") => 'c/d'
95
+ #
96
+ # Also the %d, %p, %f, %n, %x, and %X operators can take a
97
+ # pattern/replacement argument to perform simple string substitutions on a
98
+ # particular part of the path. The pattern and replacement are separated
99
+ # by a comma and are enclosed by curly braces. The replacement spec comes
100
+ # after the % character but before the operator letter. (e.g.
101
+ # "%{old,new}d"). Multiple replacement specs should be separated by
102
+ # semi-colons (e.g. "%{old,new;src,bin}d").
103
+ #
104
+ # Regular expressions may be used for the pattern, and back refs may be
105
+ # used in the replacement text. Curly braces, commas and semi-colons are
106
+ # excluded from both the pattern and replacement text (let's keep parsing
107
+ # reasonable).
108
+ #
109
+ # For example:
110
+ #
111
+ # "src/org/onestepback/proj/A.java".pathmap("%{^src,bin}X.class")
112
+ #
113
+ # returns:
114
+ #
115
+ # "bin/org/onestepback/proj/A.class"
116
+ #
117
+ # If the replacement text is '*', then a block may be provided to perform
118
+ # some arbitrary calculation for the replacement.
119
+ #
120
+ # For example:
121
+ #
122
+ # "/path/to/file.TXT".pathmap("%X%{.*,*}x") { |ext|
123
+ # ext.downcase
124
+ # }
125
+ #
126
+ # Returns:
127
+ #
128
+ # "/path/to/file.txt"
129
+ #
130
+ def pathmap(spec=nil, &block)
131
+ return self if spec.nil?
132
+ result = ''
133
+ spec.scan(/%\{[^}]*\}-?\d*[sdpfnxX%]|%-?\d+d|%.|[^%]+/) do |frag|
134
+ case frag
135
+ when '%f'
136
+ result << File.basename(self)
137
+ when '%n'
138
+ result << File.basename(self).ext
139
+ when '%d'
140
+ result << File.dirname(self)
141
+ when '%x'
142
+ result << File.extname(self)
143
+ when '%X'
144
+ result << self.ext
145
+ when '%p'
146
+ result << self
147
+ when '%s'
148
+ result << (File::ALT_SEPARATOR || File::SEPARATOR)
149
+ when '%-'
150
+ # do nothing
151
+ when '%%'
152
+ result << "%"
153
+ when /%(-?\d+)d/
154
+ result << pathmap_partial($1.to_i)
155
+ when /^%\{([^}]*)\}(\d*[dpfnxX])/
156
+ patterns, operator = $1, $2
157
+ result << pathmap('%' + operator).pathmap_replace(patterns, &block)
158
+ when /^%/
159
+ fail ArgumentError, "Unknown pathmap specifier #{frag} in '#{spec}'"
160
+ else
161
+ result << frag
162
+ end
163
+ end
164
+ result
165
+ end
166
+ end
167
+
168
+ end
@@ -0,0 +1,15 @@
1
+ #--
2
+ # Extensions to time to allow comparisons with an early time class.
3
+
4
+ require 'rake/early_time'
5
+
6
+ class Time
7
+ alias rake_original_time_compare :<=>
8
+ def <=>(other)
9
+ if Rake::EarlyTime === other
10
+ - other.<=>(self)
11
+ else
12
+ rake_original_time_compare(other)
13
+ end
14
+ end
15
+ end
@@ -0,0 +1,24 @@
1
+ require 'rake/file_task'
2
+ require 'rake/early_time'
3
+
4
+ module Rake
5
+
6
+ # A FileCreationTask is a file task that when used as a dependency will be
7
+ # needed if and only if the file has not been created. Once created, it is
8
+ # not re-triggered if any of its dependencies are newer, nor does trigger
9
+ # any rebuilds of tasks that depend on it whenever it is updated.
10
+ #
11
+ class FileCreationTask < FileTask
12
+ # Is this file task needed? Yes if it doesn't exist.
13
+ def needed?
14
+ ! File.exist?(name)
15
+ end
16
+
17
+ # Time stamp for file creation task. This time stamp is earlier
18
+ # than any other time stamp.
19
+ def timestamp
20
+ Rake::EARLY
21
+ end
22
+ end
23
+
24
+ end
@@ -0,0 +1,410 @@
1
+ require 'rake/cloneable'
2
+ require 'rake/file_utils_ext'
3
+ require 'rake/pathmap'
4
+
5
+ ######################################################################
6
+ module Rake
7
+
8
+ # #########################################################################
9
+ # A FileList is essentially an array with a few helper methods defined to
10
+ # make file manipulation a bit easier.
11
+ #
12
+ # FileLists are lazy. When given a list of glob patterns for possible files
13
+ # to be included in the file list, instead of searching the file structures
14
+ # to find the files, a FileList holds the pattern for latter use.
15
+ #
16
+ # This allows us to define a number of FileList to match any number of
17
+ # files, but only search out the actual files when then FileList itself is
18
+ # actually used. The key is that the first time an element of the
19
+ # FileList/Array is requested, the pending patterns are resolved into a real
20
+ # list of file names.
21
+ #
22
+ class FileList
23
+
24
+ include Cloneable
25
+
26
+ # == Method Delegation
27
+ #
28
+ # The lazy evaluation magic of FileLists happens by implementing all the
29
+ # array specific methods to call +resolve+ before delegating the heavy
30
+ # lifting to an embedded array object (@items).
31
+ #
32
+ # In addition, there are two kinds of delegation calls. The regular kind
33
+ # delegates to the @items array and returns the result directly. Well,
34
+ # almost directly. It checks if the returned value is the @items object
35
+ # itself, and if so will return the FileList object instead.
36
+ #
37
+ # The second kind of delegation call is used in methods that normally
38
+ # return a new Array object. We want to capture the return value of these
39
+ # methods and wrap them in a new FileList object. We enumerate these
40
+ # methods in the +SPECIAL_RETURN+ list below.
41
+
42
+ # List of array methods (that are not in +Object+) that need to be
43
+ # delegated.
44
+ ARRAY_METHODS = (Array.instance_methods - Object.instance_methods).map { |n| n.to_s }
45
+
46
+ # List of additional methods that must be delegated.
47
+ MUST_DEFINE = %w[to_a inspect <=>]
48
+
49
+ # List of methods that should not be delegated here (we define special
50
+ # versions of them explicitly below).
51
+ MUST_NOT_DEFINE = %w[to_a to_ary partition *]
52
+
53
+ # List of delegated methods that return new array values which need
54
+ # wrapping.
55
+ SPECIAL_RETURN = %w[
56
+ map collect sort sort_by select find_all reject grep
57
+ compact flatten uniq values_at
58
+ + - & |
59
+ ]
60
+
61
+ DELEGATING_METHODS = (ARRAY_METHODS + MUST_DEFINE - MUST_NOT_DEFINE).collect{ |s| s.to_s }.sort.uniq
62
+
63
+ # Now do the delegation.
64
+ DELEGATING_METHODS.each_with_index do |sym, i|
65
+ if SPECIAL_RETURN.include?(sym)
66
+ ln = __LINE__+1
67
+ class_eval %{
68
+ def #{sym}(*args, &block)
69
+ resolve
70
+ result = @items.send(:#{sym}, *args, &block)
71
+ FileList.new.import(result)
72
+ end
73
+ }, __FILE__, ln
74
+ else
75
+ ln = __LINE__+1
76
+ class_eval %{
77
+ def #{sym}(*args, &block)
78
+ resolve
79
+ result = @items.send(:#{sym}, *args, &block)
80
+ result.object_id == @items.object_id ? self : result
81
+ end
82
+ }, __FILE__, ln
83
+ end
84
+ end
85
+
86
+ # Create a file list from the globbable patterns given. If you wish to
87
+ # perform multiple includes or excludes at object build time, use the
88
+ # "yield self" pattern.
89
+ #
90
+ # Example:
91
+ # file_list = FileList.new('lib/**/*.rb', 'test/test*.rb')
92
+ #
93
+ # pkg_files = FileList.new('lib/**/*') do |fl|
94
+ # fl.exclude(/\bCVS\b/)
95
+ # end
96
+ #
97
+ def initialize(*patterns)
98
+ @pending_add = []
99
+ @pending = false
100
+ @exclude_patterns = DEFAULT_IGNORE_PATTERNS.dup
101
+ @exclude_procs = DEFAULT_IGNORE_PROCS.dup
102
+ @items = []
103
+ patterns.each { |pattern| include(pattern) }
104
+ yield self if block_given?
105
+ end
106
+
107
+ # Add file names defined by glob patterns to the file list. If an array
108
+ # is given, add each element of the array.
109
+ #
110
+ # Example:
111
+ # file_list.include("*.java", "*.cfg")
112
+ # file_list.include %w( math.c lib.h *.o )
113
+ #
114
+ def include(*filenames)
115
+ # TODO: check for pending
116
+ filenames.each do |fn|
117
+ if fn.respond_to? :to_ary
118
+ include(*fn.to_ary)
119
+ else
120
+ @pending_add << fn
121
+ end
122
+ end
123
+ @pending = true
124
+ self
125
+ end
126
+ alias :add :include
127
+
128
+ # Register a list of file name patterns that should be excluded from the
129
+ # list. Patterns may be regular expressions, glob patterns or regular
130
+ # strings. In addition, a block given to exclude will remove entries that
131
+ # return true when given to the block.
132
+ #
133
+ # Note that glob patterns are expanded against the file system. If a file
134
+ # is explicitly added to a file list, but does not exist in the file
135
+ # system, then an glob pattern in the exclude list will not exclude the
136
+ # file.
137
+ #
138
+ # Examples:
139
+ # FileList['a.c', 'b.c'].exclude("a.c") => ['b.c']
140
+ # FileList['a.c', 'b.c'].exclude(/^a/) => ['b.c']
141
+ #
142
+ # If "a.c" is a file, then ...
143
+ # FileList['a.c', 'b.c'].exclude("a.*") => ['b.c']
144
+ #
145
+ # If "a.c" is not a file, then ...
146
+ # FileList['a.c', 'b.c'].exclude("a.*") => ['a.c', 'b.c']
147
+ #
148
+ def exclude(*patterns, &block)
149
+ patterns.each do |pat|
150
+ @exclude_patterns << pat
151
+ end
152
+ if block_given?
153
+ @exclude_procs << block
154
+ end
155
+ resolve_exclude if ! @pending
156
+ self
157
+ end
158
+
159
+
160
+ # Clear all the exclude patterns so that we exclude nothing.
161
+ def clear_exclude
162
+ @exclude_patterns = []
163
+ @exclude_procs = []
164
+ self
165
+ end
166
+
167
+ # Define equality.
168
+ def ==(array)
169
+ to_ary == array
170
+ end
171
+
172
+ # Return the internal array object.
173
+ def to_a
174
+ resolve
175
+ @items
176
+ end
177
+
178
+ # Return the internal array object.
179
+ def to_ary
180
+ to_a
181
+ end
182
+
183
+ # Lie about our class.
184
+ def is_a?(klass)
185
+ klass == Array || super(klass)
186
+ end
187
+ alias kind_of? is_a?
188
+
189
+ # Redefine * to return either a string or a new file list.
190
+ def *(other)
191
+ result = @items * other
192
+ case result
193
+ when Array
194
+ FileList.new.import(result)
195
+ else
196
+ result
197
+ end
198
+ end
199
+
200
+ # Resolve all the pending adds now.
201
+ def resolve
202
+ if @pending
203
+ @pending = false
204
+ @pending_add.each do |fn| resolve_add(fn) end
205
+ @pending_add = []
206
+ resolve_exclude
207
+ end
208
+ self
209
+ end
210
+
211
+ def resolve_add(fn)
212
+ case fn
213
+ when %r{[*?\[\{]}
214
+ add_matching(fn)
215
+ else
216
+ self << fn
217
+ end
218
+ end
219
+ private :resolve_add
220
+
221
+ def resolve_exclude
222
+ reject! { |fn| exclude?(fn) }
223
+ self
224
+ end
225
+ private :resolve_exclude
226
+
227
+ # Return a new FileList with the results of running +sub+ against each
228
+ # element of the original list.
229
+ #
230
+ # Example:
231
+ # FileList['a.c', 'b.c'].sub(/\.c$/, '.o') => ['a.o', 'b.o']
232
+ #
233
+ def sub(pat, rep)
234
+ inject(FileList.new) { |res, fn| res << fn.sub(pat,rep) }
235
+ end
236
+
237
+ # Return a new FileList with the results of running +gsub+ against each
238
+ # element of the original list.
239
+ #
240
+ # Example:
241
+ # FileList['lib/test/file', 'x/y'].gsub(/\//, "\\")
242
+ # => ['lib\\test\\file', 'x\\y']
243
+ #
244
+ def gsub(pat, rep)
245
+ inject(FileList.new) { |res, fn| res << fn.gsub(pat,rep) }
246
+ end
247
+
248
+ # Same as +sub+ except that the original file list is modified.
249
+ def sub!(pat, rep)
250
+ each_with_index { |fn, i| self[i] = fn.sub(pat,rep) }
251
+ self
252
+ end
253
+
254
+ # Same as +gsub+ except that the original file list is modified.
255
+ def gsub!(pat, rep)
256
+ each_with_index { |fn, i| self[i] = fn.gsub(pat,rep) }
257
+ self
258
+ end
259
+
260
+ # Apply the pathmap spec to each of the included file names, returning a
261
+ # new file list with the modified paths. (See String#pathmap for
262
+ # details.)
263
+ def pathmap(spec=nil)
264
+ collect { |fn| fn.pathmap(spec) }
265
+ end
266
+
267
+ # Return a new FileList with <tt>String#ext</tt> method applied to
268
+ # each member of the array.
269
+ #
270
+ # This method is a shortcut for:
271
+ #
272
+ # array.collect { |item| item.ext(newext) }
273
+ #
274
+ # +ext+ is a user added method for the Array class.
275
+ def ext(newext='')
276
+ collect { |fn| fn.ext(newext) }
277
+ end
278
+
279
+
280
+ # Grep each of the files in the filelist using the given pattern. If a
281
+ # block is given, call the block on each matching line, passing the file
282
+ # name, line number, and the matching line of text. If no block is given,
283
+ # a standard emacs style file:linenumber:line message will be printed to
284
+ # standard out. Returns the number of matched items.
285
+ def egrep(pattern, *options)
286
+ matched = 0
287
+ each do |fn|
288
+ begin
289
+ open(fn, "r", *options) do |inf|
290
+ count = 0
291
+ inf.each do |line|
292
+ count += 1
293
+ if pattern.match(line)
294
+ matched += 1
295
+ if block_given?
296
+ yield fn, count, line
297
+ else
298
+ puts "#{fn}:#{count}:#{line}"
299
+ end
300
+ end
301
+ end
302
+ end
303
+ rescue StandardError => ex
304
+ $stderr.puts "Error while processing '#{fn}': #{ex}"
305
+ end
306
+ end
307
+ matched
308
+ end
309
+
310
+ # Return a new file list that only contains file names from the current
311
+ # file list that exist on the file system.
312
+ def existing
313
+ select { |fn| File.exist?(fn) }
314
+ end
315
+
316
+ # Modify the current file list so that it contains only file name that
317
+ # exist on the file system.
318
+ def existing!
319
+ resolve
320
+ @items = @items.select { |fn| File.exist?(fn) }
321
+ self
322
+ end
323
+
324
+ # FileList version of partition. Needed because the nested arrays should
325
+ # be FileLists in this version.
326
+ def partition(&block) # :nodoc:
327
+ resolve
328
+ result = @items.partition(&block)
329
+ [
330
+ FileList.new.import(result[0]),
331
+ FileList.new.import(result[1]),
332
+ ]
333
+ end
334
+
335
+ # Convert a FileList to a string by joining all elements with a space.
336
+ def to_s
337
+ resolve
338
+ self.join(' ')
339
+ end
340
+
341
+ # Add matching glob patterns.
342
+ def add_matching(pattern)
343
+ FileList.glob(pattern).each do |fn|
344
+ self << fn unless exclude?(fn)
345
+ end
346
+ end
347
+ private :add_matching
348
+
349
+ # Should the given file name be excluded?
350
+ def exclude?(fn)
351
+ return true if @exclude_patterns.any? do |pat|
352
+ case pat
353
+ when Regexp
354
+ fn =~ pat
355
+ when /[*?]/
356
+ File.fnmatch?(pat, fn, File::FNM_PATHNAME)
357
+ else
358
+ fn == pat
359
+ end
360
+ end
361
+ @exclude_procs.any? { |p| p.call(fn) }
362
+ end
363
+
364
+ DEFAULT_IGNORE_PATTERNS = [
365
+ /(^|[\/\\])CVS([\/\\]|$)/,
366
+ /(^|[\/\\])\.svn([\/\\]|$)/,
367
+ /\.bak$/,
368
+ /~$/
369
+ ]
370
+ DEFAULT_IGNORE_PROCS = [
371
+ proc { |fn| fn =~ /(^|[\/\\])core$/ && ! File.directory?(fn) }
372
+ ]
373
+
374
+ def import(array)
375
+ @items = array
376
+ self
377
+ end
378
+
379
+ class << self
380
+ # Create a new file list including the files listed. Similar to:
381
+ #
382
+ # FileList.new(*args)
383
+ def [](*args)
384
+ new(*args)
385
+ end
386
+
387
+ # Get a sorted list of files matching the pattern. This method
388
+ # should be prefered to Dir[pattern] and Dir.glob(pattern) because
389
+ # the files returned are guaranteed to be sorted.
390
+ def glob(pattern, *args)
391
+ Dir.glob(pattern, *args).sort
392
+ end
393
+ end
394
+ end
395
+ end
396
+
397
+ module Rake
398
+ class << self
399
+
400
+ # Yield each file or directory component.
401
+ def each_dir_parent(dir) # :nodoc:
402
+ old_length = nil
403
+ while dir != '.' && dir.length != old_length
404
+ yield(dir)
405
+ old_length = dir.length
406
+ dir = File.dirname(dir)
407
+ end
408
+ end
409
+ end
410
+ end # module Rake