thor-plus 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,148 @@
1
+ class Thor
2
+ class Option < Argument #:nodoc:
3
+ attr_reader :aliases, :group, :lazy_default, :hide, :incremental
4
+
5
+ VALID_TYPES = [:boolean, :numeric, :hash, :array, :string]
6
+
7
+ def initialize(name, options = {})
8
+ options[:required] = false unless options.key?(:required)
9
+ super
10
+ @lazy_default = options[:lazy_default]
11
+ @group = options[:group].to_s.capitalize if options[:group]
12
+ @aliases = Array(options[:aliases])
13
+ @hide = options[:hide]
14
+ @incremental = options[:incremental]
15
+ end
16
+
17
+ # Is this numeric option incremental?
18
+ #
19
+ # e.g. -vvv = 3
20
+ def incremental?
21
+ numeric? && incremental
22
+ end
23
+
24
+ # This parse quick options given as method_options. It makes several
25
+ # assumptions, but you can be more specific using the option method.
26
+ #
27
+ # parse :foo => "bar"
28
+ # #=> Option foo with default value bar
29
+ #
30
+ # parse [:foo, :baz] => "bar"
31
+ # #=> Option foo with default value bar and alias :baz
32
+ #
33
+ # parse :foo => :required
34
+ # #=> Required option foo without default value
35
+ #
36
+ # parse :foo => 2
37
+ # #=> Option foo with default value 2 and type numeric
38
+ #
39
+ # parse :foo => :numeric
40
+ # #=> Option foo without default value and type numeric
41
+ #
42
+ # parse :foo => true
43
+ # #=> Option foo with default value true and type boolean
44
+ #
45
+ # The valid types are :boolean, :numeric, :hash, :array and :string. If none
46
+ # is given a default type is assumed. This default type accepts arguments as
47
+ # string (--foo=value) or booleans (just --foo).
48
+ #
49
+ # By default all options are optional, unless :required is given.
50
+ #
51
+ def self.parse(key, value) # rubocop:disable MethodLength
52
+ if key.is_a?(Array)
53
+ name, *aliases = key
54
+ else
55
+ name, aliases = key, []
56
+ end
57
+
58
+ name = name.to_s
59
+ default = value
60
+
61
+ type = case value
62
+ when Symbol
63
+ default = nil
64
+ if VALID_TYPES.include?(value)
65
+ value
66
+ elsif required = (value == :required) # rubocop:disable AssignmentInCondition
67
+ :string
68
+ end
69
+ when TrueClass, FalseClass
70
+ :boolean
71
+ when Numeric
72
+ :numeric
73
+ when Hash, Array, String
74
+ value.class.name.downcase.to_sym
75
+ end
76
+ new(name.to_s, :required => required, :type => type, :default => default, :aliases => aliases)
77
+ end
78
+
79
+ def switch_name
80
+ @switch_name ||= dasherized? ? name : dasherize(name)
81
+ end
82
+
83
+ def human_name
84
+ @human_name ||= dasherized? ? undasherize(name) : name
85
+ end
86
+
87
+ def usage(padding = 0)
88
+ sample = if banner && !banner.to_s.empty?
89
+ "#{switch_name}=#{banner}"
90
+ else
91
+ switch_name
92
+ end
93
+
94
+ sample = "[#{sample}]" unless required?
95
+
96
+ if boolean?
97
+ sample << ", [#{dasherize("no-" + human_name)}]" unless name == "force" or name.start_with?("no-")
98
+ end
99
+
100
+ if aliases.empty?
101
+ (" " * padding) << sample
102
+ else
103
+ "#{aliases.join(', ')}, #{sample}"
104
+ end
105
+ end
106
+
107
+ VALID_TYPES.each do |type|
108
+ class_eval <<-RUBY, __FILE__, __LINE__ + 1
109
+ def #{type}?
110
+ self.type == #{type.inspect}
111
+ end
112
+ RUBY
113
+ end
114
+
115
+ protected
116
+
117
+ def validate!
118
+ fail ArgumentError, "An option cannot be boolean and required." if boolean? && required?
119
+ validate_default_type!
120
+ end
121
+
122
+ def validate_default_type!
123
+ default_type = case @default
124
+ when nil
125
+ return
126
+ when TrueClass, FalseClass
127
+ :boolean
128
+ when Numeric
129
+ :numeric
130
+ when Hash, Array, String
131
+ @default.class.name.downcase.to_sym
132
+ end
133
+ fail ArgumentError, "An option's default must match its type." unless default_type == @type
134
+ end
135
+
136
+ def dasherized?
137
+ name.index("-") == 0
138
+ end
139
+
140
+ def undasherize(str)
141
+ str.sub(/^-{1,2}/, "")
142
+ end
143
+
144
+ def dasherize(str)
145
+ (str.length > 1 ? "--" : "-") + str.gsub("_", "-")
146
+ end
147
+ end
148
+ end
@@ -0,0 +1,228 @@
1
+ class Thor
2
+ class Options < Arguments #:nodoc: # rubocop:disable ClassLength
3
+ LONG_RE = /^(--\w+(?:-\w+)*)$/
4
+ SHORT_RE = /^(-[a-z])$/i
5
+ EQ_RE = /^(--\w+(?:-\w+)*|-[a-z])=(.*)$/i
6
+ SHORT_SQ_RE = /^-([a-z]{2,})$/i # Allow either -x -v or -xv style for single char args
7
+ SHORT_NUM = /^(-[a-z])#{NUMERIC}$/i
8
+ SHORT_INC = /^-((.)\2{1,})$/i # Allow the same character repeated for an incremental arg.
9
+ OPTS_END = "--".freeze
10
+
11
+ # Receives a hash and makes it switches.
12
+ def self.to_switches(options)
13
+ options.map do |key, value|
14
+ case value
15
+ when true
16
+ "--#{key}"
17
+ when Array
18
+ "--#{key} #{value.map { |v| v.inspect }.join(' ')}"
19
+ when Hash
20
+ "--#{key} #{value.map { |k, v| "#{k}:#{v}" }.join(' ')}"
21
+ when nil, false
22
+ ""
23
+ else
24
+ "--#{key} #{value.inspect}"
25
+ end
26
+ end.join(" ")
27
+ end
28
+
29
+ # Takes a hash of Thor::Option and a hash with defaults.
30
+ #
31
+ # If +stop_on_unknown+ is true, #parse will stop as soon as it encounters
32
+ # an unknown option or a regular argument.
33
+ def initialize(hash_options = {}, defaults = {}, stop_on_unknown = false)
34
+ @stop_on_unknown = stop_on_unknown
35
+ options = hash_options.values
36
+ super(options)
37
+
38
+ # Add defaults
39
+ defaults.each do |key, value|
40
+ @assigns[key.to_s] = value
41
+ @non_assigned_required.delete(hash_options[key])
42
+ end
43
+
44
+ @shorts, @switches, @extra = {}, {}, []
45
+
46
+ options.each do |option|
47
+ @switches[option.switch_name] = option
48
+
49
+ option.aliases.each do |short|
50
+ name = short.to_s.sub(/^(?!\-)/, "-")
51
+ @shorts[name] ||= option.switch_name
52
+ end
53
+ end
54
+ end
55
+
56
+ def remaining # rubocop:disable TrivialAccessors
57
+ @extra
58
+ end
59
+
60
+ def peek
61
+ return super unless @parsing_options
62
+
63
+ result = super
64
+ if result == OPTS_END
65
+ shift
66
+ @parsing_options = false
67
+ super
68
+ else
69
+ result
70
+ end
71
+ end
72
+
73
+ def parse(args) # rubocop:disable MethodLength
74
+ @pile = args.dup
75
+ @parsing_options = true
76
+
77
+ while peek
78
+ if parsing_options?
79
+ match, is_switch = current_is_switch?
80
+ shifted = shift
81
+
82
+ if is_switch
83
+ case shifted
84
+ when SHORT_INC
85
+ # The numeric value of this switch is the length of the repeated short switches.
86
+ unshift($1.length)
87
+ switch = "-#{$2}"
88
+ when SHORT_SQ_RE
89
+ unshift($1.split("").map { |f| "-#{f}" })
90
+ next
91
+ when EQ_RE, SHORT_NUM
92
+ unshift($2)
93
+ switch = $1
94
+ when LONG_RE, SHORT_RE
95
+ switch = $1
96
+ end
97
+
98
+ switch = normalize_switch(switch)
99
+ option = switch_option(switch)
100
+ @assigns[option.human_name] = parse_peek(switch, option)
101
+ elsif @stop_on_unknown
102
+ @parsing_options = false
103
+ @extra << shifted
104
+ @extra << shift while peek
105
+ break
106
+ elsif match
107
+ @extra << shifted
108
+ @extra << shift while peek && peek !~ /^-/
109
+ else
110
+ @extra << shifted
111
+ end
112
+ else
113
+ @extra << shift
114
+ end
115
+ end
116
+
117
+ check_requirement!
118
+
119
+ assigns = Thor::CoreExt::HashWithIndifferentAccess.new(@assigns)
120
+ assigns.freeze
121
+ assigns
122
+ end
123
+
124
+ def check_unknown!
125
+ # an unknown option starts with - or -- and has no more --'s afterward.
126
+ unknown = @extra.select { |str| str =~ /^--?(?:(?!--).)*$/ }
127
+ fail UnknownArgumentError, "Unknown switches '#{unknown.join(', ')}'" unless unknown.empty?
128
+ end
129
+
130
+ protected
131
+
132
+ # Check if the current value in peek is a registered switch.
133
+ #
134
+ # Two booleans are returned. The first is true if the current value
135
+ # starts with a hyphen; the second is true if it is a registered switch.
136
+ def current_is_switch?
137
+ case peek
138
+ when LONG_RE, SHORT_RE, EQ_RE, SHORT_NUM
139
+ [true, switch?($1)]
140
+ when SHORT_INC
141
+ [false, switch?("-#{$2}")]
142
+ when SHORT_SQ_RE
143
+ [true, $1.split("").any? { |f| switch?("-#{f}") }]
144
+ else
145
+ [false, false]
146
+ end
147
+ end
148
+
149
+ def current_is_switch_formatted?
150
+ case peek
151
+ when LONG_RE, SHORT_RE, EQ_RE, SHORT_NUM, SHORT_SQ_RE
152
+ true
153
+ else
154
+ false
155
+ end
156
+ end
157
+
158
+ def current_is_value?
159
+ peek && (!parsing_options? || super)
160
+ end
161
+
162
+ def switch?(arg)
163
+ switch_option(normalize_switch(arg))
164
+ end
165
+
166
+ def switch_option(arg)
167
+ if match = no_or_skip?(arg) # rubocop:disable AssignmentInCondition
168
+ @switches[arg] || @switches["--#{match}"]
169
+ else
170
+ @switches[arg]
171
+ end
172
+ end
173
+
174
+ # Check if the given argument is actually a shortcut.
175
+ #
176
+ def normalize_switch(arg)
177
+ (@shorts[arg] || arg).tr("_", "-")
178
+ end
179
+
180
+ def parsing_options?
181
+ peek
182
+ @parsing_options
183
+ end
184
+
185
+ # Parse boolean values which can be given as --foo=true, --foo or --no-foo.
186
+ #
187
+ def parse_boolean(switch)
188
+ if current_is_value?
189
+ if ["true", "TRUE", "t", "T", true].include?(peek)
190
+ shift
191
+ true
192
+ elsif ["false", "FALSE", "f", "F", false].include?(peek)
193
+ shift
194
+ false
195
+ else
196
+ true
197
+ end
198
+ else
199
+ @switches.key?(switch) || !no_or_skip?(switch)
200
+ end
201
+ end
202
+
203
+ # Parse the value at the peek analyzing if it requires an input or not.
204
+ #
205
+ def parse_peek(switch, option)
206
+ if parsing_options? && (current_is_switch_formatted? || last?)
207
+ if option.boolean?
208
+ # No problem for boolean types
209
+ elsif option.numeric? && option.incremental?
210
+ # If this is an incremental numeric option and no value is provided, default to 1.
211
+ return 1
212
+ elsif no_or_skip?(switch)
213
+ return nil # User set value to nil
214
+ elsif option.string? && !option.required?
215
+ # Return the default if there is one, else the human name
216
+ return option.lazy_default || option.default || option.human_name
217
+ elsif option.lazy_default
218
+ return option.lazy_default
219
+ else
220
+ fail MalformattedArgumentError, "No value provided for option '#{switch}'"
221
+ end
222
+ end
223
+
224
+ @non_assigned_required.delete(option)
225
+ send(:"parse_#{option.type}", switch)
226
+ end
227
+ end
228
+ end
@@ -0,0 +1,71 @@
1
+ require "rake"
2
+ require "rake/dsl_definition"
3
+
4
+ class Thor
5
+ # Adds a compatibility layer to your Thor classes which allows you to use
6
+ # rake package tasks. For example, to use rspec rake tasks, one can do:
7
+ #
8
+ # require 'thor/rake_compat'
9
+ # require 'rspec/core/rake_task'
10
+ #
11
+ # class Default < Thor
12
+ # include Thor::RakeCompat
13
+ #
14
+ # RSpec::Core::RakeTask.new(:spec) do |t|
15
+ # t.spec_opts = ['--options', './.rspec']
16
+ # t.spec_files = FileList['spec/**/*_spec.rb']
17
+ # end
18
+ # end
19
+ #
20
+ module RakeCompat
21
+ include Rake::DSL if defined?(Rake::DSL)
22
+
23
+ def self.rake_classes
24
+ @rake_classes ||= []
25
+ end
26
+
27
+ def self.included(base)
28
+ # Hack. Make rakefile point to invoker, so rdoc task is generated properly.
29
+ rakefile = File.basename(caller[0].match(/(.*):\d+/)[1])
30
+ Rake.application.instance_variable_set(:@rakefile, rakefile)
31
+ rake_classes << base
32
+ end
33
+ end
34
+ end
35
+
36
+ # override task on (main), for compatibility with Rake 0.9
37
+ instance_eval do
38
+ alias rake_namespace namespace
39
+
40
+ def task(*)
41
+ task = super
42
+
43
+ if klass = Thor::RakeCompat.rake_classes.last # rubocop:disable AssignmentInCondition
44
+ non_namespaced_name = task.name.split(":").last
45
+
46
+ description = non_namespaced_name
47
+ description << task.arg_names.map { |n| n.to_s.upcase }.join(" ")
48
+ description.strip!
49
+
50
+ klass.desc description, Rake.application.last_description || non_namespaced_name
51
+ Rake.application.last_description = nil
52
+ klass.send :define_method, non_namespaced_name do |*args|
53
+ Rake::Task[task.name.to_sym].invoke(*args)
54
+ end
55
+ end
56
+
57
+ task
58
+ end
59
+
60
+ def namespace(name)
61
+ if klass = Thor::RakeCompat.rake_classes.last # rubocop:disable AssignmentInCondition
62
+ const_name = Thor::Util.camel_case(name.to_s).to_sym
63
+ klass.const_set(const_name, Class.new(Thor))
64
+ new_klass = klass.const_get(const_name)
65
+ Thor::RakeCompat.rake_classes << new_klass
66
+ end
67
+
68
+ super
69
+ Thor::RakeCompat.rake_classes.pop
70
+ end
71
+ end
@@ -0,0 +1,322 @@
1
+ require "thor"
2
+ require "thor/group"
3
+ require "thor/core_ext/io_binary_read"
4
+
5
+ require "fileutils"
6
+ require "open-uri"
7
+ require "yaml"
8
+ require "digest/md5"
9
+ require "pathname"
10
+
11
+ class Thor::Runner < Thor #:nodoc: # rubocop:disable ClassLength
12
+ map "-T" => :list, "-i" => :install, "-u" => :update, "-v" => :version
13
+
14
+ # Override Thor#help so it can give information about any class and any method.
15
+ #
16
+ def help(meth = nil)
17
+ if meth && !self.respond_to?(meth)
18
+ initialize_thorfiles(meth)
19
+ klass, command = Thor::Util.find_class_and_command_by_namespace(meth)
20
+ self.class.handle_no_command_error(command, false) if klass.nil?
21
+ klass.start(["-h", command].compact, :shell => shell)
22
+ else
23
+ super
24
+ end
25
+ end
26
+
27
+ # If a command is not found on Thor::Runner, method missing is invoked and
28
+ # Thor::Runner is then responsible for finding the command in all classes.
29
+ #
30
+ def method_missing(meth, *args)
31
+ meth = meth.to_s
32
+ initialize_thorfiles(meth)
33
+ klass, command = Thor::Util.find_class_and_command_by_namespace(meth)
34
+ self.class.handle_no_command_error(command, false) if klass.nil?
35
+ args.unshift(command) if command
36
+ klass.start(args, :shell => shell)
37
+ end
38
+
39
+ desc "install NAME", "Install an optionally named Thor file into your system commands"
40
+ method_options :as => :string, :relative => :boolean, :force => :boolean
41
+ def install(name) # rubocop:disable MethodLength
42
+ initialize_thorfiles
43
+
44
+ # If a directory name is provided as the argument, look for a 'main.thor'
45
+ # command in said directory.
46
+ begin
47
+ if File.directory?(File.expand_path(name))
48
+ base, package = File.join(name, "main.thor"), :directory
49
+ contents = open(base) { |input| input.read }
50
+ else
51
+ base, package = name, :file
52
+ contents = open(name) { |input| input.read }
53
+ end
54
+ rescue OpenURI::HTTPError
55
+ raise Error, "Error opening URI '#{name}'"
56
+ rescue Errno::ENOENT
57
+ fail Error, "Error opening file '#{name}'"
58
+ end
59
+
60
+ say "Your Thorfile contains:"
61
+ say contents
62
+
63
+ unless options["force"]
64
+ return false if no?("Do you wish to continue [y/N]?")
65
+ end
66
+
67
+ as = options["as"] || begin
68
+ first_line = contents.split("\n")[0]
69
+ (match = first_line.match(/\s*#\s*module:\s*([^\n]*)/)) ? match[1].strip : nil
70
+ end
71
+
72
+ unless as
73
+ basename = File.basename(name)
74
+ as = ask("Please specify a name for #{name} in the system repository [#{basename}]:")
75
+ as = basename if as.empty?
76
+ end
77
+
78
+ location = if options[:relative] || name =~ %r{^https?://}
79
+ name
80
+ else
81
+ File.expand_path(name)
82
+ end
83
+
84
+ thor_yaml[as] = {
85
+ :filename => Digest::MD5.hexdigest(name + as),
86
+ :location => location,
87
+ :namespaces => Thor::Util.namespaces_in_content(contents, base)
88
+ }
89
+
90
+ save_yaml(thor_yaml)
91
+ say "Storing thor file in your system repository"
92
+ destination = File.join(thor_root, thor_yaml[as][:filename])
93
+
94
+ if package == :file
95
+ File.open(destination, "w") { |f| f.puts contents }
96
+ else
97
+ FileUtils.cp_r(name, destination)
98
+ end
99
+
100
+ thor_yaml[as][:filename] # Indicate success
101
+ end
102
+
103
+ desc "version", "Show Thor version"
104
+ def version
105
+ require "thor/version"
106
+ say "Thor #{Thor::VERSION}"
107
+ end
108
+
109
+ desc "uninstall NAME", "Uninstall a named Thor module"
110
+ def uninstall(name)
111
+ fail Error, "Can't find module '#{name}'" unless thor_yaml[name]
112
+ say "Uninstalling #{name}."
113
+ FileUtils.rm_rf(File.join(thor_root, "#{thor_yaml[name][:filename]}"))
114
+
115
+ thor_yaml.delete(name)
116
+ save_yaml(thor_yaml)
117
+
118
+ puts "Done."
119
+ end
120
+
121
+ desc "update NAME", "Update a Thor file from its original location"
122
+ def update(name)
123
+ fail Error, "Can't find module '#{name}'" if !thor_yaml[name] || !thor_yaml[name][:location]
124
+
125
+ say "Updating '#{name}' from #{thor_yaml[name][:location]}"
126
+
127
+ old_filename = thor_yaml[name][:filename]
128
+ self.options = options.merge("as" => name)
129
+
130
+ if File.directory? File.expand_path(name)
131
+ FileUtils.rm_rf(File.join(thor_root, old_filename))
132
+
133
+ thor_yaml.delete(old_filename)
134
+ save_yaml(thor_yaml)
135
+
136
+ filename = install(name)
137
+ else
138
+ filename = install(thor_yaml[name][:location])
139
+ end
140
+
141
+ unless filename == old_filename
142
+ File.delete(File.join(thor_root, old_filename))
143
+ end
144
+ end
145
+
146
+ desc "installed", "List the installed Thor modules and commands"
147
+ method_options :internal => :boolean
148
+ def installed
149
+ initialize_thorfiles(nil, true)
150
+ display_klasses(true, options["internal"])
151
+ end
152
+
153
+ desc "list [SEARCH]", "List the available thor commands (--substring means .*SEARCH)"
154
+ method_options :substring => :boolean, :group => :string, :all => :boolean, :debug => :boolean
155
+ def list(search = "")
156
+ initialize_thorfiles
157
+
158
+ search = ".*#{search}" if options["substring"]
159
+ search = /^#{search}.*/i
160
+ group = options[:group] || "standard"
161
+
162
+ klasses = Thor::Base.subclasses.select do |k|
163
+ (options[:all] || k.group == group) && k.namespace =~ search
164
+ end
165
+
166
+ display_klasses(false, false, klasses)
167
+ end
168
+
169
+ private
170
+
171
+ def self.banner(command, all = false, subcommand = false)
172
+ "thor " + command.formatted_usage(self, all, subcommand)
173
+ end
174
+
175
+ def thor_root
176
+ Thor::Util.thor_root
177
+ end
178
+
179
+ def thor_yaml
180
+ @thor_yaml ||= begin
181
+ yaml_file = File.join(thor_root, "thor.yml")
182
+ yaml = YAML.load_file(yaml_file) if File.exist?(yaml_file)
183
+ yaml || {}
184
+ end
185
+ end
186
+
187
+ # Save the yaml file. If none exists in thor root, creates one.
188
+ #
189
+ def save_yaml(yaml)
190
+ yaml_file = File.join(thor_root, "thor.yml")
191
+
192
+ unless File.exist?(yaml_file)
193
+ FileUtils.mkdir_p(thor_root)
194
+ yaml_file = File.join(thor_root, "thor.yml")
195
+ FileUtils.touch(yaml_file)
196
+ end
197
+
198
+ File.open(yaml_file, "w") { |f| f.puts yaml.to_yaml }
199
+ end
200
+
201
+ def self.exit_on_failure?
202
+ true
203
+ end
204
+
205
+ # Load the Thorfiles. If relevant_to is supplied, looks for specific files
206
+ # in the thor_root instead of loading them all.
207
+ #
208
+ # By default, it also traverses the current path until find Thor files, as
209
+ # described in thorfiles. This look up can be skipped by supplying
210
+ # skip_lookup true.
211
+ #
212
+ def initialize_thorfiles(relevant_to = nil, skip_lookup = false)
213
+ thorfiles(relevant_to, skip_lookup).each do |f|
214
+ Thor::Util.load_thorfile(f, nil, options[:debug]) unless Thor::Base.subclass_files.keys.include?(File.expand_path(f))
215
+ end
216
+ end
217
+
218
+ # Finds Thorfiles by traversing from your current directory down to the root
219
+ # directory of your system. If at any time we find a Thor file, we stop.
220
+ #
221
+ # We also ensure that system-wide Thorfiles are loaded first, so local
222
+ # Thorfiles can override them.
223
+ #
224
+ # ==== Example
225
+ #
226
+ # If we start at /Users/wycats/dev/thor ...
227
+ #
228
+ # 1. /Users/wycats/dev/thor
229
+ # 2. /Users/wycats/dev
230
+ # 3. /Users/wycats <-- we find a Thorfile here, so we stop
231
+ #
232
+ # Suppose we start at c:\Documents and Settings\james\dev\thor ...
233
+ #
234
+ # 1. c:\Documents and Settings\james\dev\thor
235
+ # 2. c:\Documents and Settings\james\dev
236
+ # 3. c:\Documents and Settings\james
237
+ # 4. c:\Documents and Settings
238
+ # 5. c:\ <-- no Thorfiles found!
239
+ #
240
+ def thorfiles(relevant_to = nil, skip_lookup = false)
241
+ thorfiles = []
242
+
243
+ unless skip_lookup
244
+ Pathname.pwd.ascend do |path|
245
+ thorfiles = Thor::Util.globs_for(path).map { |g| Dir[g] }.flatten
246
+ break unless thorfiles.empty?
247
+ end
248
+ end
249
+
250
+ files = (relevant_to ? thorfiles_relevant_to(relevant_to) : Thor::Util.thor_root_glob)
251
+ files += thorfiles
252
+ files -= ["#{thor_root}/thor.yml"]
253
+
254
+ files.map! do |file|
255
+ File.directory?(file) ? File.join(file, "main.thor") : file
256
+ end
257
+ end
258
+
259
+ # Load Thorfiles relevant to the given method. If you provide "foo:bar" it
260
+ # will load all thor files in the thor.yaml that has "foo" e "foo:bar"
261
+ # namespaces registered.
262
+ #
263
+ def thorfiles_relevant_to(meth)
264
+ lookup = [meth, meth.split(":")[0...-1].join(":")]
265
+
266
+ files = thor_yaml.select do |k, v|
267
+ v[:namespaces] && !(v[:namespaces] & lookup).empty?
268
+ end
269
+
270
+ files.map { |k, v| File.join(thor_root, "#{v[:filename]}") }
271
+ end
272
+
273
+ # Display information about the given klasses. If with_module is given,
274
+ # it shows a table with information extracted from the yaml file.
275
+ #
276
+ def display_klasses(with_modules = false, show_internal = false, klasses = Thor::Base.subclasses)
277
+ klasses -= [Thor, Thor::Runner, Thor::Group] unless show_internal
278
+
279
+ fail Error, "No Thor commands available" if klasses.empty?
280
+ show_modules if with_modules && !thor_yaml.empty?
281
+
282
+ list = Hash.new { |h, k| h[k] = [] }
283
+ groups = klasses.select { |k| k.ancestors.include?(Thor::Group) }
284
+
285
+ # Get classes which inherit from Thor
286
+ (klasses - groups).each { |k| list[k.namespace.split(":").first] += k.printable_commands(false) }
287
+
288
+ # Get classes which inherit from Thor::Base
289
+ groups.map! { |k| k.printable_commands(false).first }
290
+ list["root"] = groups
291
+
292
+ # Order namespaces with default coming first
293
+ list = list.sort { |a, b| a[0].sub(/^default/, "") <=> b[0].sub(/^default/, "") }
294
+ list.each { |n, commands| display_commands(n, commands) unless commands.empty? }
295
+ end
296
+
297
+ def display_commands(namespace, list) #:nodoc:
298
+ list.sort! { |a, b| a[0] <=> b[0] }
299
+
300
+ say shell.set_color(namespace, :blue, true)
301
+ say "-" * namespace.size
302
+
303
+ print_table(list, :truncate => true)
304
+ say
305
+ end
306
+ alias_method :display_tasks, :display_commands
307
+
308
+ def show_modules #:nodoc:
309
+ info = []
310
+ labels = %w[Modules Namespaces]
311
+
312
+ info << labels
313
+ info << ["-" * labels[0].size, "-" * labels[1].size]
314
+
315
+ thor_yaml.each do |name, hash|
316
+ info << [name, hash[:namespaces].join(", ")]
317
+ end
318
+
319
+ print_table info
320
+ say ""
321
+ end
322
+ end