atli 0.1.2

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.
Files changed (42) hide show
  1. checksums.yaml +7 -0
  2. data/.yardopts +8 -0
  3. data/CHANGELOG.md +193 -0
  4. data/CONTRIBUTING.md +20 -0
  5. data/LICENSE.md +24 -0
  6. data/README.md +44 -0
  7. data/atli.gemspec +30 -0
  8. data/bin/thor +6 -0
  9. data/lib/thor.rb +868 -0
  10. data/lib/thor/actions.rb +322 -0
  11. data/lib/thor/actions/create_file.rb +104 -0
  12. data/lib/thor/actions/create_link.rb +60 -0
  13. data/lib/thor/actions/directory.rb +118 -0
  14. data/lib/thor/actions/empty_directory.rb +143 -0
  15. data/lib/thor/actions/file_manipulation.rb +364 -0
  16. data/lib/thor/actions/inject_into_file.rb +109 -0
  17. data/lib/thor/base.rb +773 -0
  18. data/lib/thor/command.rb +192 -0
  19. data/lib/thor/core_ext/hash_with_indifferent_access.rb +97 -0
  20. data/lib/thor/core_ext/io_binary_read.rb +12 -0
  21. data/lib/thor/core_ext/ordered_hash.rb +129 -0
  22. data/lib/thor/error.rb +32 -0
  23. data/lib/thor/group.rb +281 -0
  24. data/lib/thor/invocation.rb +182 -0
  25. data/lib/thor/line_editor.rb +17 -0
  26. data/lib/thor/line_editor/basic.rb +37 -0
  27. data/lib/thor/line_editor/readline.rb +88 -0
  28. data/lib/thor/parser.rb +5 -0
  29. data/lib/thor/parser/argument.rb +70 -0
  30. data/lib/thor/parser/arguments.rb +175 -0
  31. data/lib/thor/parser/option.rb +146 -0
  32. data/lib/thor/parser/options.rb +221 -0
  33. data/lib/thor/parser/shared_option.rb +23 -0
  34. data/lib/thor/rake_compat.rb +71 -0
  35. data/lib/thor/runner.rb +324 -0
  36. data/lib/thor/shell.rb +81 -0
  37. data/lib/thor/shell/basic.rb +439 -0
  38. data/lib/thor/shell/color.rb +149 -0
  39. data/lib/thor/shell/html.rb +126 -0
  40. data/lib/thor/util.rb +268 -0
  41. data/lib/thor/version.rb +22 -0
  42. metadata +114 -0
@@ -0,0 +1,221 @@
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
+ OPTS_END = "--".freeze
9
+
10
+ # Receives a hash and makes it switches.
11
+ def self.to_switches(options)
12
+ options.map do |key, value|
13
+ case value
14
+ when true
15
+ "--#{key}"
16
+ when Array
17
+ "--#{key} #{value.map(&:inspect).join(' ')}"
18
+ when Hash
19
+ "--#{key} #{value.map { |k, v| "#{k}:#{v}" }.join(' ')}"
20
+ when nil, false
21
+ nil
22
+ else
23
+ "--#{key} #{value.inspect}"
24
+ end
25
+ end.compact.join(" ")
26
+ end
27
+
28
+ # Takes a hash of Thor::Option and a hash with defaults.
29
+ #
30
+ # If +stop_on_unknown+ is true, #parse will stop as soon as it encounters
31
+ # an unknown option or a regular argument.
32
+ def initialize(hash_options = {}, defaults = {}, stop_on_unknown = false, disable_required_check = false)
33
+ @stop_on_unknown = stop_on_unknown
34
+ @disable_required_check = disable_required_check
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 = {}
45
+ @switches = {}
46
+ @extra = []
47
+
48
+ options.each do |option|
49
+ @switches[option.switch_name] = option
50
+
51
+ option.aliases.each do |short|
52
+ name = short.to_s.sub(/^(?!\-)/, "-")
53
+ @shorts[name] ||= option.switch_name
54
+ end
55
+ end
56
+ end
57
+
58
+ def remaining
59
+ @extra
60
+ end
61
+
62
+ def peek
63
+ return super unless @parsing_options
64
+
65
+ result = super
66
+ if result == OPTS_END
67
+ shift
68
+ @parsing_options = false
69
+ super
70
+ else
71
+ result
72
+ end
73
+ end
74
+
75
+ def parse(args) # rubocop:disable MethodLength
76
+ @pile = args.dup
77
+ @parsing_options = true
78
+
79
+ while peek
80
+ if parsing_options?
81
+ match, is_switch = current_is_switch?
82
+ shifted = shift
83
+
84
+ if is_switch
85
+ case shifted
86
+ when SHORT_SQ_RE
87
+ unshift($1.split("").map { |f| "-#{f}" })
88
+ next
89
+ when EQ_RE, SHORT_NUM
90
+ unshift($2)
91
+ switch = $1
92
+ when LONG_RE, SHORT_RE
93
+ switch = $1
94
+ end
95
+
96
+ switch = normalize_switch(switch)
97
+ option = switch_option(switch)
98
+ @assigns[option.human_name] = parse_peek(switch, option)
99
+ elsif @stop_on_unknown
100
+ @parsing_options = false
101
+ @extra << shifted
102
+ @extra << shift while peek
103
+ break
104
+ elsif match
105
+ @extra << shifted
106
+ @extra << shift while peek && peek !~ /^-/
107
+ else
108
+ @extra << shifted
109
+ end
110
+ else
111
+ @extra << shift
112
+ end
113
+ end
114
+
115
+ check_requirement! unless @disable_required_check
116
+
117
+ assigns = Thor::CoreExt::HashWithIndifferentAccess.new(@assigns)
118
+ assigns.freeze
119
+ assigns
120
+ end
121
+
122
+ def check_unknown!
123
+ # an unknown option starts with - or -- and has no more --'s afterward.
124
+ unknown = @extra.select { |str| str =~ /^--?(?:(?!--).)*$/ }
125
+ raise UnknownArgumentError, "Unknown switches '#{unknown.join(', ')}'" unless unknown.empty?
126
+ end
127
+
128
+ protected
129
+
130
+ # Check if the current value in peek is a registered switch.
131
+ #
132
+ # Two booleans are returned. The first is true if the current value
133
+ # starts with a hyphen; the second is true if it is a registered switch.
134
+ def current_is_switch?
135
+ case peek
136
+ when LONG_RE, SHORT_RE, EQ_RE, SHORT_NUM
137
+ [true, switch?($1)]
138
+ when SHORT_SQ_RE
139
+ [true, $1.split("").any? { |f| switch?("-#{f}") }]
140
+ else
141
+ [false, false]
142
+ end
143
+ end
144
+
145
+ def current_is_switch_formatted?
146
+ case peek
147
+ when LONG_RE, SHORT_RE, EQ_RE, SHORT_NUM, SHORT_SQ_RE
148
+ true
149
+ else
150
+ false
151
+ end
152
+ end
153
+
154
+ def current_is_value?
155
+ peek && (!parsing_options? || super)
156
+ end
157
+
158
+ def switch?(arg)
159
+ switch_option(normalize_switch(arg))
160
+ end
161
+
162
+ def switch_option(arg)
163
+ if match = no_or_skip?(arg) # rubocop:disable AssignmentInCondition
164
+ @switches[arg] || @switches["--#{match}"]
165
+ else
166
+ @switches[arg]
167
+ end
168
+ end
169
+
170
+ # Check if the given argument is actually a shortcut.
171
+ #
172
+ def normalize_switch(arg)
173
+ (@shorts[arg] || arg).tr("_", "-")
174
+ end
175
+
176
+ def parsing_options?
177
+ peek
178
+ @parsing_options
179
+ end
180
+
181
+ # Parse boolean values which can be given as --foo=true, --foo or --no-foo.
182
+ #
183
+ def parse_boolean(switch)
184
+ if current_is_value?
185
+ if ["true", "TRUE", "t", "T", true].include?(peek)
186
+ shift
187
+ true
188
+ elsif ["false", "FALSE", "f", "F", false].include?(peek)
189
+ shift
190
+ false
191
+ else
192
+ !no_or_skip?(switch)
193
+ end
194
+ else
195
+ @switches.key?(switch) || !no_or_skip?(switch)
196
+ end
197
+ end
198
+
199
+ # Parse the value at the peek analyzing if it requires an input or not.
200
+ #
201
+ def parse_peek(switch, option)
202
+ if parsing_options? && (current_is_switch_formatted? || last?)
203
+ if option.boolean?
204
+ # No problem for boolean types
205
+ elsif no_or_skip?(switch)
206
+ return nil # User set value to nil
207
+ elsif option.string? && !option.required?
208
+ # Return the default if there is one, else the human name
209
+ return option.lazy_default || option.default || option.human_name
210
+ elsif option.lazy_default
211
+ return option.lazy_default
212
+ else
213
+ raise MalformattedArgumentError, "No value provided for option '#{switch}'"
214
+ end
215
+ end
216
+
217
+ @non_assigned_required.delete(option)
218
+ send(:"parse_#{option.type}", switch)
219
+ end
220
+ end
221
+ end
@@ -0,0 +1,23 @@
1
+ require 'set'
2
+
3
+ class Thor
4
+ # A {Thor::Option} that has an additional {#groups} attribute storing a
5
+ # set of group symbols that the option is a part of.
6
+ #
7
+ class SharedOption < Option
8
+
9
+ # Shared option groups this option belongs to.
10
+ #
11
+ # @return [Set<Symbol>]
12
+ #
13
+ attr_reader :groups
14
+
15
+ #
16
+ #
17
+ def initialize name, **options
18
+ super name, options
19
+
20
+ @groups = Set.new [*options[:groups]].map( &:to_sym )
21
+ end
22
+ end
23
+ 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,324 @@
1
+ require "thor"
2
+ require "thor/group"
3
+ require "thor/core_ext/io_binary_read"
4
+
5
+ require "yaml"
6
+ require "digest/md5"
7
+ require "pathname"
8
+
9
+ class Thor::Runner < Thor #:nodoc: # rubocop:disable ClassLength
10
+ map "-T" => :list, "-i" => :install, "-u" => :update, "-v" => :version
11
+
12
+ def self.banner(command, all = false, subcommand = false)
13
+ "thor " + command.formatted_usage(self, all, subcommand)
14
+ end
15
+
16
+ def self.exit_on_failure?
17
+ true
18
+ end
19
+
20
+ # Override Thor#help so it can give information about any class and any method.
21
+ #
22
+ def help(meth = nil)
23
+ if meth && !respond_to?(meth)
24
+ initialize_thorfiles(meth)
25
+ klass, command = Thor::Util.find_class_and_command_by_namespace(meth)
26
+ self.class.handle_no_command_error(command, false) if klass.nil?
27
+ klass.start(["-h", command].compact, :shell => shell)
28
+ else
29
+ super
30
+ end
31
+ end
32
+
33
+ # If a command is not found on Thor::Runner, method missing is invoked and
34
+ # Thor::Runner is then responsible for finding the command in all classes.
35
+ #
36
+ def method_missing(meth, *args)
37
+ meth = meth.to_s
38
+ initialize_thorfiles(meth)
39
+ klass, command = Thor::Util.find_class_and_command_by_namespace(meth)
40
+ self.class.handle_no_command_error(command, false) if klass.nil?
41
+ args.unshift(command) if command
42
+ klass.start(args, :shell => shell)
43
+ end
44
+
45
+ desc "install NAME", "Install an optionally named Thor file into your system commands"
46
+ method_options :as => :string, :relative => :boolean, :force => :boolean
47
+ def install(name) # rubocop:disable MethodLength
48
+ initialize_thorfiles
49
+
50
+ # If a directory name is provided as the argument, look for a 'main.thor'
51
+ # command in said directory.
52
+ begin
53
+ if File.directory?(File.expand_path(name))
54
+ base = File.join(name, "main.thor")
55
+ package = :directory
56
+ contents = open(base, &:read)
57
+ else
58
+ base = name
59
+ package = :file
60
+ contents = open(name, &:read)
61
+ end
62
+ rescue OpenURI::HTTPError
63
+ raise Error, "Error opening URI '#{name}'"
64
+ rescue Errno::ENOENT
65
+ raise Error, "Error opening file '#{name}'"
66
+ end
67
+
68
+ say "Your Thorfile contains:"
69
+ say contents
70
+
71
+ unless options["force"]
72
+ return false if no?("Do you wish to continue [y/N]?")
73
+ end
74
+
75
+ as = options["as"] || begin
76
+ first_line = contents.split("\n")[0]
77
+ (match = first_line.match(/\s*#\s*module:\s*([^\n]*)/)) ? match[1].strip : nil
78
+ end
79
+
80
+ unless as
81
+ basename = File.basename(name)
82
+ as = ask("Please specify a name for #{name} in the system repository [#{basename}]:")
83
+ as = basename if as.empty?
84
+ end
85
+
86
+ location = if options[:relative] || name =~ %r{^https?://}
87
+ name
88
+ else
89
+ File.expand_path(name)
90
+ end
91
+
92
+ thor_yaml[as] = {
93
+ :filename => Digest::MD5.hexdigest(name + as),
94
+ :location => location,
95
+ :namespaces => Thor::Util.namespaces_in_content(contents, base)
96
+ }
97
+
98
+ save_yaml(thor_yaml)
99
+ say "Storing thor file in your system repository"
100
+ destination = File.join(thor_root, thor_yaml[as][:filename])
101
+
102
+ if package == :file
103
+ File.open(destination, "w") { |f| f.puts contents }
104
+ else
105
+ require "fileutils"
106
+ FileUtils.cp_r(name, destination)
107
+ end
108
+
109
+ thor_yaml[as][:filename] # Indicate success
110
+ end
111
+
112
+ desc "version", "Show Thor version"
113
+ def version
114
+ require "thor/version"
115
+ say "Thor #{Thor::VERSION}"
116
+ end
117
+
118
+ desc "uninstall NAME", "Uninstall a named Thor module"
119
+ def uninstall(name)
120
+ raise Error, "Can't find module '#{name}'" unless thor_yaml[name]
121
+ say "Uninstalling #{name}."
122
+ require "fileutils"
123
+ FileUtils.rm_rf(File.join(thor_root, (thor_yaml[name][:filename]).to_s))
124
+
125
+ thor_yaml.delete(name)
126
+ save_yaml(thor_yaml)
127
+
128
+ puts "Done."
129
+ end
130
+
131
+ desc "update NAME", "Update a Thor file from its original location"
132
+ def update(name)
133
+ raise Error, "Can't find module '#{name}'" if !thor_yaml[name] || !thor_yaml[name][:location]
134
+
135
+ say "Updating '#{name}' from #{thor_yaml[name][:location]}"
136
+
137
+ old_filename = thor_yaml[name][:filename]
138
+ self.options = options.merge("as" => name)
139
+
140
+ if File.directory? File.expand_path(name)
141
+ require "fileutils"
142
+ FileUtils.rm_rf(File.join(thor_root, old_filename))
143
+
144
+ thor_yaml.delete(old_filename)
145
+ save_yaml(thor_yaml)
146
+
147
+ filename = install(name)
148
+ else
149
+ filename = install(thor_yaml[name][:location])
150
+ end
151
+
152
+ File.delete(File.join(thor_root, old_filename)) unless filename == old_filename
153
+ end
154
+
155
+ desc "installed", "List the installed Thor modules and commands"
156
+ method_options :internal => :boolean
157
+ def installed
158
+ initialize_thorfiles(nil, true)
159
+ display_klasses(true, options["internal"])
160
+ end
161
+
162
+ desc "list [SEARCH]", "List the available thor commands (--substring means .*SEARCH)"
163
+ method_options :substring => :boolean, :group => :string, :all => :boolean, :debug => :boolean
164
+ def list(search = "")
165
+ initialize_thorfiles
166
+
167
+ search = ".*#{search}" if options["substring"]
168
+ search = /^#{search}.*/i
169
+ group = options[:group] || "standard"
170
+
171
+ klasses = Thor::Base.subclasses.select do |k|
172
+ (options[:all] || k.group == group) && k.namespace =~ search
173
+ end
174
+
175
+ display_klasses(false, false, klasses)
176
+ end
177
+
178
+ private
179
+
180
+ def thor_root
181
+ Thor::Util.thor_root
182
+ end
183
+
184
+ def thor_yaml
185
+ @thor_yaml ||= begin
186
+ yaml_file = File.join(thor_root, "thor.yml")
187
+ yaml = YAML.load_file(yaml_file) if File.exist?(yaml_file)
188
+ yaml || {}
189
+ end
190
+ end
191
+
192
+ # Save the yaml file. If none exists in thor root, creates one.
193
+ #
194
+ def save_yaml(yaml)
195
+ yaml_file = File.join(thor_root, "thor.yml")
196
+
197
+ unless File.exist?(yaml_file)
198
+ require "fileutils"
199
+ FileUtils.mkdir_p(thor_root)
200
+ yaml_file = File.join(thor_root, "thor.yml")
201
+ FileUtils.touch(yaml_file)
202
+ end
203
+
204
+ File.open(yaml_file, "w") { |f| f.puts yaml.to_yaml }
205
+ end
206
+
207
+ # Load the Thorfiles. If relevant_to is supplied, looks for specific files
208
+ # in the thor_root instead of loading them all.
209
+ #
210
+ # By default, it also traverses the current path until find Thor files, as
211
+ # described in thorfiles. This look up can be skipped by supplying
212
+ # skip_lookup true.
213
+ #
214
+ def initialize_thorfiles(relevant_to = nil, skip_lookup = false)
215
+ thorfiles(relevant_to, skip_lookup).each do |f|
216
+ Thor::Util.load_thorfile(f, nil, options[:debug]) unless Thor::Base.subclass_files.keys.include?(File.expand_path(f))
217
+ end
218
+ end
219
+
220
+ # Finds Thorfiles by traversing from your current directory down to the root
221
+ # directory of your system. If at any time we find a Thor file, we stop.
222
+ #
223
+ # We also ensure that system-wide Thorfiles are loaded first, so local
224
+ # Thorfiles can override them.
225
+ #
226
+ # ==== Example
227
+ #
228
+ # If we start at /Users/wycats/dev/thor ...
229
+ #
230
+ # 1. /Users/wycats/dev/thor
231
+ # 2. /Users/wycats/dev
232
+ # 3. /Users/wycats <-- we find a Thorfile here, so we stop
233
+ #
234
+ # Suppose we start at c:\Documents and Settings\james\dev\thor ...
235
+ #
236
+ # 1. c:\Documents and Settings\james\dev\thor
237
+ # 2. c:\Documents and Settings\james\dev
238
+ # 3. c:\Documents and Settings\james
239
+ # 4. c:\Documents and Settings
240
+ # 5. c:\ <-- no Thorfiles found!
241
+ #
242
+ def thorfiles(relevant_to = nil, skip_lookup = false)
243
+ thorfiles = []
244
+
245
+ unless skip_lookup
246
+ Pathname.pwd.ascend do |path|
247
+ thorfiles = Thor::Util.globs_for(path).map { |g| Dir[g] }.flatten
248
+ break unless thorfiles.empty?
249
+ end
250
+ end
251
+
252
+ files = (relevant_to ? thorfiles_relevant_to(relevant_to) : Thor::Util.thor_root_glob)
253
+ files += thorfiles
254
+ files -= ["#{thor_root}/thor.yml"]
255
+
256
+ files.map! do |file|
257
+ File.directory?(file) ? File.join(file, "main.thor") : file
258
+ end
259
+ end
260
+
261
+ # Load Thorfiles relevant to the given method. If you provide "foo:bar" it
262
+ # will load all thor files in the thor.yaml that has "foo" e "foo:bar"
263
+ # namespaces registered.
264
+ #
265
+ def thorfiles_relevant_to(meth)
266
+ lookup = [meth, meth.split(":")[0...-1].join(":")]
267
+
268
+ files = thor_yaml.select do |_, v|
269
+ v[:namespaces] && !(v[:namespaces] & lookup).empty?
270
+ end
271
+
272
+ files.map { |_, v| File.join(thor_root, (v[:filename]).to_s) }
273
+ end
274
+
275
+ # Display information about the given klasses. If with_module is given,
276
+ # it shows a table with information extracted from the yaml file.
277
+ #
278
+ def display_klasses(with_modules = false, show_internal = false, klasses = Thor::Base.subclasses)
279
+ klasses -= [Thor, Thor::Runner, Thor::Group] unless show_internal
280
+
281
+ raise Error, "No Thor commands available" if klasses.empty?
282
+ show_modules if with_modules && !thor_yaml.empty?
283
+
284
+ list = Hash.new { |h, k| h[k] = [] }
285
+ groups = klasses.select { |k| k.ancestors.include?(Thor::Group) }
286
+
287
+ # Get classes which inherit from Thor
288
+ (klasses - groups).each { |k| list[k.namespace.split(":").first] += k.printable_commands(false) }
289
+
290
+ # Get classes which inherit from Thor::Base
291
+ groups.map! { |k| k.printable_commands(false).first }
292
+ list["root"] = groups
293
+
294
+ # Order namespaces with default coming first
295
+ list = list.sort { |a, b| a[0].sub(/^default/, "") <=> b[0].sub(/^default/, "") }
296
+ list.each { |n, commands| display_commands(n, commands) unless commands.empty? }
297
+ end
298
+
299
+ def display_commands(namespace, list) #:nodoc:
300
+ list.sort! { |a, b| a[0] <=> b[0] }
301
+
302
+ say shell.set_color(namespace, :blue, true)
303
+ say "-" * namespace.size
304
+
305
+ print_table(list, :truncate => true)
306
+ say
307
+ end
308
+ alias_method :display_tasks, :display_commands
309
+
310
+ def show_modules #:nodoc:
311
+ info = []
312
+ labels = %w(Modules Namespaces)
313
+
314
+ info << labels
315
+ info << ["-" * labels[0].size, "-" * labels[1].size]
316
+
317
+ thor_yaml.each do |name, hash|
318
+ info << [name, hash[:namespaces].join(", ")]
319
+ end
320
+
321
+ print_table info
322
+ say ""
323
+ end
324
+ end