command-set 0.8.4 → 0.9.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.
- data/doc/argumentDSL +2 -0
- data/lib/command-set/arguments.rb +94 -80
- data/lib/command-set/command-common.rb +109 -0
- data/lib/command-set/command-set.rb +57 -58
- data/lib/command-set/command.rb +124 -118
- data/lib/command-set/dsl.rb +180 -100
- data/lib/command-set/interpreter.rb +11 -2
- data/lib/command-set/quick-interpreter.rb +9 -1
- data/lib/command-set/results.rb +2 -0
- data/lib/command-set/standard-commands.rb +2 -11
- data/lib/command-set/subject.rb +11 -7
- data/lib/command-set/text-interpreter.rb +3 -3
- metadata +4 -4
| @@ -1,10 +1,12 @@ | |
| 1 1 | 
             
            require 'strscan'
         | 
| 2 2 | 
             
            require 'thread'
         | 
| 3 | 
            +
            require 'forwardable'
         | 
| 3 4 | 
             
            require 'command-set/arguments'
         | 
| 4 5 | 
             
            require 'command-set/command'
         | 
| 5 6 | 
             
            require 'command-set/subject'
         | 
| 6 7 | 
             
            require 'command-set/results'
         | 
| 7 8 | 
             
            require 'command-set/dsl'
         | 
| 9 | 
            +
            require 'command-set/command-common'
         | 
| 8 10 |  | 
| 9 11 |  | 
| 10 12 | 
             
            #:stopdoc:
         | 
| @@ -122,13 +124,24 @@ module Command | |
| 122 124 | 
             
              class ArgumentInvalidException < CommandException
         | 
| 123 125 | 
             
                def initialize(pairs)
         | 
| 124 126 | 
             
                  @pairs = pairs.dup
         | 
| 125 | 
            -
                  super(" | 
| 127 | 
            +
                  super("Invalid arguments: #{pairs.map{|n,v| "#{n}: #{v.inspect}"}.join(", ")}")
         | 
| 126 128 | 
             
                end
         | 
| 127 129 |  | 
| 128 130 | 
             
                attr_reader :pairs
         | 
| 129 131 | 
             
              end
         | 
| 130 132 | 
             
              class ArgumentUnrecognizedException < CommandException; end
         | 
| 131 133 |  | 
| 134 | 
            +
              class RootCommand < Command
         | 
| 135 | 
            +
                class << self
         | 
| 136 | 
            +
                  def single_consume(argument, arg_hash, subject, terms)
         | 
| 137 | 
            +
                    if not argument.required? and parent.command_names.include?(terms.first)
         | 
| 138 | 
            +
                      raise OutOfArgumentsException, "next is a command name"
         | 
| 139 | 
            +
                    end
         | 
| 140 | 
            +
                    super
         | 
| 141 | 
            +
                  end
         | 
| 142 | 
            +
                end
         | 
| 143 | 
            +
              end
         | 
| 144 | 
            +
             | 
| 132 145 | 
             
              #This class packs up a set of commands, for presentation to an
         | 
| 133 146 | 
             
              #interpreter.  CommandSet objects are defined using methods from
         | 
| 134 147 | 
             
              #DSL::CommandSetDefinition
         | 
| @@ -136,13 +149,15 @@ module Command | |
| 136 149 | 
             
                def initialize(parent=nil, name="")
         | 
| 137 150 | 
             
                  @parent = parent
         | 
| 138 151 | 
             
                  @name = name
         | 
| 139 | 
            -
                  @command_list =  | 
| 152 | 
            +
                  @command_list = { nil => RootCommand.setup(self, nil) {} }
         | 
| 140 153 | 
             
                  @subject_template = nil
         | 
| 141 154 | 
             
                  @documentation = ""
         | 
| 142 155 | 
             
                  @prompt = nil
         | 
| 156 | 
            +
                  @arguments = []
         | 
| 157 | 
            +
                  @most_recent_args = {}
         | 
| 143 158 | 
             
                end
         | 
| 144 159 |  | 
| 145 | 
            -
                attr_accessor :documentation, :parent
         | 
| 160 | 
            +
                attr_accessor :documentation, :parent, :most_recent_args
         | 
| 146 161 |  | 
| 147 162 | 
             
                class << self
         | 
| 148 163 | 
             
                  #The preferred way to use a CommandSet is to call CommandSet::define_commands with
         | 
| @@ -173,9 +188,44 @@ module Command | |
| 173 188 | 
             
                end
         | 
| 174 189 |  | 
| 175 190 | 
             
                include DSL::CommandSetDefinition
         | 
| 191 | 
            +
                include Common
         | 
| 192 | 
            +
                extend Forwardable
         | 
| 193 | 
            +
             | 
| 194 | 
            +
                def_delegators("@command_list[nil]", *CommandClassMethods.instance_methods)
         | 
| 195 | 
            +
                def_delegators("@command_list[nil]", *DSL::CommandDefinition.instance_methods)
         | 
| 176 196 |  | 
| 177 197 | 
             
                #:section: Workhorse methods - not usually used by client code
         | 
| 178 198 | 
             
                #
         | 
| 199 | 
            +
                def each_command(&block)
         | 
| 200 | 
            +
                  @command_list.each_value do |command|
         | 
| 201 | 
            +
                    command.each_command(&block)
         | 
| 202 | 
            +
                  end
         | 
| 203 | 
            +
                end
         | 
| 204 | 
            +
             | 
| 205 | 
            +
                def visit(terms, visitor)
         | 
| 206 | 
            +
                  if(terms.empty?)
         | 
| 207 | 
            +
                    return visitor.set_out_of_terms(self)
         | 
| 208 | 
            +
                  end
         | 
| 209 | 
            +
             | 
| 210 | 
            +
                  next_hop = @command_list[terms.first]
         | 
| 211 | 
            +
                  if(next_hop.nil?)
         | 
| 212 | 
            +
                    return visitor.term_without_hop(terms, self)
         | 
| 213 | 
            +
                  else
         | 
| 214 | 
            +
                    terms.shift
         | 
| 215 | 
            +
                  end
         | 
| 216 | 
            +
             | 
| 217 | 
            +
                  visitor.leave_from(terms, self)
         | 
| 218 | 
            +
             | 
| 219 | 
            +
                  visitor.arrive_at(terms, next_hop)
         | 
| 220 | 
            +
             | 
| 221 | 
            +
                  return next_hop.visit(terms, visitor)
         | 
| 222 | 
            +
                end
         | 
| 223 | 
            +
             | 
| 224 | 
            +
                def consume_terms(terms, subject, arg_hash)
         | 
| 225 | 
            +
                  @command_list[nil].consume_terms(terms, subject, arg_hash)
         | 
| 226 | 
            +
                  @most_recent_args = arg_hash.dup
         | 
| 227 | 
            +
                  return arg_hash
         | 
| 228 | 
            +
                end
         | 
| 179 229 |  | 
| 180 230 | 
             
                def get_root
         | 
| 181 231 | 
             
                  command = @command_list[nil]
         | 
| @@ -204,61 +254,6 @@ module Command | |
| 204 254 | 
             
                  return subject
         | 
| 205 255 | 
             
                end
         | 
| 206 256 |  | 
| 207 | 
            -
                #Given a set of terms, returns a Command or CommandSet, plus an extraneous (probably argument) terms
         | 
| 208 | 
            -
                def find_command(*terms)
         | 
| 209 | 
            -
                  if terms.empty?
         | 
| 210 | 
            -
                    return self,[] 
         | 
| 211 | 
            -
                  end
         | 
| 212 | 
            -
             | 
| 213 | 
            -
                  command_name = terms.shift
         | 
| 214 | 
            -
             | 
| 215 | 
            -
                  command = @command_list[command_name]
         | 
| 216 | 
            -
             | 
| 217 | 
            -
                  if Class === command && Command > command
         | 
| 218 | 
            -
            	if command.defined?
         | 
| 219 | 
            -
            	  return command,terms
         | 
| 220 | 
            -
            	else
         | 
| 221 | 
            -
            	  raise CommandError, "#{command_name}: unfinalized command: #{command.inspect}"
         | 
| 222 | 
            -
            	end
         | 
| 223 | 
            -
                  elsif CommandSet === command
         | 
| 224 | 
            -
            	return command.find_command(*terms)
         | 
| 225 | 
            -
                  elsif command.nil?
         | 
| 226 | 
            -
                    if @command_list[nil].nil?
         | 
| 227 | 
            -
                      raise CommandException, "Unknown command: #{command_name.inspect}"
         | 
| 228 | 
            -
                    else
         | 
| 229 | 
            -
                      return @command_list[nil], terms.unshift(command_name)
         | 
| 230 | 
            -
                    end
         | 
| 231 | 
            -
                  else
         | 
| 232 | 
            -
            	raise ScriptError, "Somehow a non-command made it's " +
         | 
| 233 | 
            -
            	  "way into the command registry. -- (#{command.inspect})"
         | 
| 234 | 
            -
                  end
         | 
| 235 | 
            -
                end
         | 
| 236 | 
            -
             | 
| 237 | 
            -
                def completion_list(terms, prefix, subject)
         | 
| 238 | 
            -
                  if terms.length == 0
         | 
| 239 | 
            -
            	list = @command_list.keys.grep(%r{^#{prefix}.*})
         | 
| 240 | 
            -
            	if @command_list.has_key?(nil)
         | 
| 241 | 
            -
            	  list += @command_list[nil].completion_list([], prefix, subject)
         | 
| 242 | 
            -
            	end
         | 
| 243 | 
            -
            	return list
         | 
| 244 | 
            -
                  end
         | 
| 245 | 
            -
             | 
| 246 | 
            -
                  begin
         | 
| 247 | 
            -
            	command,terms = find_command(*terms)
         | 
| 248 | 
            -
            	return command.completion_list(terms, prefix, subject)
         | 
| 249 | 
            -
                  rescue CommandException => ce
         | 
| 250 | 
            -
            	return []
         | 
| 251 | 
            -
                  end
         | 
| 252 | 
            -
                end
         | 
| 253 | 
            -
             | 
| 254 | 
            -
                def path
         | 
| 255 | 
            -
                  if @parent.nil?
         | 
| 256 | 
            -
                    return []
         | 
| 257 | 
            -
                  else
         | 
| 258 | 
            -
                    return @parent.path + [@name]
         | 
| 259 | 
            -
                  end
         | 
| 260 | 
            -
                end
         | 
| 261 | 
            -
             | 
| 262 257 | 
             
                def documentation(width, parent="")
         | 
| 263 258 | 
             
                  docs = ""
         | 
| 264 259 | 
             
                  docs = @command_list.to_a.reject do |el|
         | 
| @@ -278,6 +273,10 @@ module Command | |
| 278 273 | 
             
                def command_list
         | 
| 279 274 | 
             
                  return @command_list.dup
         | 
| 280 275 | 
             
                end
         | 
| 276 | 
            +
             | 
| 277 | 
            +
                def command_names
         | 
| 278 | 
            +
                  return @command_list.keys.compact
         | 
| 279 | 
            +
                end
         | 
| 281 280 | 
             
              end
         | 
| 282 281 | 
             
            end
         | 
| 283 282 |  | 
    
        data/lib/command-set/command.rb
    CHANGED
    
    | @@ -1,4 +1,5 @@ | |
| 1 1 | 
             
            require 'command-set/arguments'
         | 
| 2 | 
            +
            require 'command-set/command-common'
         | 
| 2 3 | 
             
            module Command
         | 
| 3 4 |  | 
| 4 5 | 
             
              #A thin wrapper on Array to maintain undo/redo state.
         | 
| @@ -46,41 +47,22 @@ module Command | |
| 46 47 |  | 
| 47 48 | 
             
                attr_accessor :task_id, :command, :args_hash, :terms
         | 
| 48 49 |  | 
| 49 | 
            -
                def resolve_command_class(command_set)
         | 
| 50 | 
            -
                   | 
| 51 | 
            -
                     | 
| 50 | 
            +
                def resolve_command_class(command_set, subject)
         | 
| 51 | 
            +
                  unless Class === @command && Command > @command
         | 
| 52 | 
            +
                    command_path = @command
         | 
| 53 | 
            +
                    result = command_set.process_terms(command_path, subject)
         | 
| 54 | 
            +
                    @command = result.command_class
         | 
| 55 | 
            +
                    @args_hash = result.arg_hash.merge(@args_hash)
         | 
| 56 | 
            +
                    @terms = command_path
         | 
| 52 57 | 
             
                  end
         | 
| 53 58 |  | 
| 54 | 
            -
                  command_path = @command
         | 
| 55 | 
            -
                  @command,terms = command_set.find_command(*command_path)
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                  if CommandSet === @command
         | 
| 58 | 
            -
                    if (cmd = @command.get_root).nil?
         | 
| 59 | 
            -
                      raise CommandException, "Incomplete command #{command_path.join(" ")}"
         | 
| 60 | 
            -
                    else
         | 
| 61 | 
            -
                      terms = [@command]
         | 
| 62 | 
            -
                      @command = cmd
         | 
| 63 | 
            -
                    end
         | 
| 64 | 
            -
                  end
         | 
| 65 | 
            -
             | 
| 66 | 
            -
                  @terms = terms
         | 
| 67 | 
            -
             | 
| 68 59 | 
             
                  return @command
         | 
| 69 60 | 
             
                end
         | 
| 70 61 |  | 
| 71 | 
            -
                def assign_terms(cmd)
         | 
| 72 | 
            -
                  terms = @terms.dup
         | 
| 73 | 
            -
                  cmd.each_consumer do |consumer|
         | 
| 74 | 
            -
                    terms = consumer[terms]
         | 
| 75 | 
            -
                  end
         | 
| 76 | 
            -
             | 
| 77 | 
            -
                  cmd.consume_hash(@args_hash)
         | 
| 78 | 
            -
                end
         | 
| 79 | 
            -
             | 
| 80 62 | 
             
                def command_instance(command_set, subject)
         | 
| 81 | 
            -
                  command_class = resolve_command_class(command_set)
         | 
| 63 | 
            +
                  command_class = resolve_command_class(command_set, subject)
         | 
| 82 64 | 
             
                  command = command_class.new(subject, task_id)
         | 
| 83 | 
            -
                   | 
| 65 | 
            +
                  command.consume_hash(@args_hash)
         | 
| 84 66 | 
             
                  return command
         | 
| 85 67 | 
             
                end
         | 
| 86 68 | 
             
              end
         | 
| @@ -99,30 +81,88 @@ module Command | |
| 99 81 |  | 
| 100 82 | 
             
              class ResumeFromOnlyThis < ResumeFrom; end
         | 
| 101 83 |  | 
| 84 | 
            +
              module CommandClassMethods
         | 
| 85 | 
            +
                attr_reader :subject_requirements, :argument_list, 
         | 
| 86 | 
            +
                  :advice_block, :parent_arguments
         | 
| 87 | 
            +
             | 
| 88 | 
            +
                def consume_terms(terms, subject, arg_hash)
         | 
| 89 | 
            +
                  begin
         | 
| 90 | 
            +
                    argument_list.each do |argument|
         | 
| 91 | 
            +
                      raise TypeError, "#{terms.inspect} isn't an Array!" unless Array === terms
         | 
| 92 | 
            +
                      begin
         | 
| 93 | 
            +
                        if terms.empty?
         | 
| 94 | 
            +
                          raise OutOfArgumentsException, "argument #{argument.inspect} required!"
         | 
| 95 | 
            +
                        end
         | 
| 96 | 
            +
                        single_consume(argument, arg_hash, subject, terms)
         | 
| 97 | 
            +
                      rescue ArgumentInvalidException => aie
         | 
| 98 | 
            +
                        @validation_problem = aie
         | 
| 99 | 
            +
                        raise
         | 
| 100 | 
            +
                      end
         | 
| 101 | 
            +
                    end
         | 
| 102 | 
            +
                  rescue OutOfArgumentsException
         | 
| 103 | 
            +
                  end
         | 
| 104 | 
            +
                  @validation_problem = nil
         | 
| 105 | 
            +
                  return arg_hash
         | 
| 106 | 
            +
                end
         | 
| 107 | 
            +
             | 
| 108 | 
            +
                def embed_argument(arg)
         | 
| 109 | 
            +
                  unless argument_list.last.nil? or argument_list.last.required?
         | 
| 110 | 
            +
                    if arg.required?
         | 
| 111 | 
            +
                      raise NoMethodError, "Can't define required arguments after optionals"
         | 
| 112 | 
            +
                    end
         | 
| 113 | 
            +
                  end
         | 
| 114 | 
            +
                  @subject_requirements += arg.subject_requirements
         | 
| 115 | 
            +
             | 
| 116 | 
            +
                  argument_list << arg
         | 
| 117 | 
            +
                end
         | 
| 118 | 
            +
             | 
| 119 | 
            +
                def executeable?
         | 
| 120 | 
            +
                  instance_methods.include?("execute")
         | 
| 121 | 
            +
                end
         | 
| 122 | 
            +
             | 
| 123 | 
            +
                def optional_argument(arg, values=nil, &get_values)
         | 
| 124 | 
            +
                  optional.argument(arg, values, &get_values)
         | 
| 125 | 
            +
                end
         | 
| 126 | 
            +
             | 
| 127 | 
            +
                def factory
         | 
| 128 | 
            +
                  return self
         | 
| 129 | 
            +
                end
         | 
| 130 | 
            +
              end
         | 
| 131 | 
            +
             | 
| 102 132 | 
             
              class Command
         | 
| 103 133 | 
             
                @name = "unnamed command"
         | 
| 104 134 | 
             
                @parent = nil
         | 
| 105 135 | 
             
                @argument_list=[]
         | 
| 136 | 
            +
                @parent_arguments=[]
         | 
| 106 137 | 
             
                @doc_text = nil
         | 
| 107 138 | 
             
                @subject_requirements=[:chain_of_command]
         | 
| 108 139 | 
             
                @defined = false
         | 
| 109 140 | 
             
                @advice_block = proc {}
         | 
| 141 | 
            +
             | 
| 110 142 | 
             
                class << self
         | 
| 111 143 | 
             
                  alias_method :instance, :new
         | 
| 112 144 |  | 
| 113 | 
            -
                  attr_reader :name, : | 
| 114 | 
            -
                  alias_method :required_argument_list, :argument_list 
         | 
| 115 | 
            -
                  alias_method :optional_argument_list, :argument_list 
         | 
| 116 | 
            -
                  attr_reader :doc_text, :subject_requirements, :defined, :advice_block
         | 
| 145 | 
            +
                  attr_reader :name, :doc_text, :defined 
         | 
| 117 146 | 
             
                  attr_accessor :parent
         | 
| 118 147 |  | 
| 119 148 | 
             
                  def inherited(subclass)
         | 
| 120 | 
            -
                     | 
| 121 | 
            -
             | 
| 122 | 
            -
             | 
| 123 | 
            -
             | 
| 124 | 
            -
             | 
| 125 | 
            -
             | 
| 149 | 
            +
                    [ [:name, proc {name.dup}],
         | 
| 150 | 
            +
                      [:argument_list, proc {argument_list.dup }],
         | 
| 151 | 
            +
                      [:parent_arguments, proc {parent_arguments.dup }],
         | 
| 152 | 
            +
                      [:doc_text, proc {doc_text.nil? ? nil : doc_text.dup}], 
         | 
| 153 | 
            +
                      [:subject_requirements, proc {subject_requirements.dup}],
         | 
| 154 | 
            +
                      [:defined, proc {false}],
         | 
| 155 | 
            +
                      [:advice_block, proc {advice_block.dup}],
         | 
| 156 | 
            +
                    ].each do |var, prok|
         | 
| 157 | 
            +
                      subclass.instance_variable_set("@#{var.to_s}", prok.call)
         | 
| 158 | 
            +
                    end
         | 
| 159 | 
            +
                  end
         | 
| 160 | 
            +
             | 
| 161 | 
            +
                  def initialize_copy(original)
         | 
| 162 | 
            +
                    super
         | 
| 163 | 
            +
                    @argument_list = original.argument_list.dup
         | 
| 164 | 
            +
                    @parent_arguments = original.parent_arguments.dup
         | 
| 165 | 
            +
                    @subject_requirements = original.subject_requirements.dup
         | 
| 126 166 | 
             
                  end
         | 
| 127 167 |  | 
| 128 168 | 
             
                  #Establishes a subclass of Command.  This is important because commands are actually
         | 
| @@ -140,8 +180,6 @@ module Command | |
| 140 180 |  | 
| 141 181 | 
             
                    command_class = Class.new(self)
         | 
| 142 182 | 
             
                    new_name = new_name.to_s
         | 
| 143 | 
            -
                    #Of interest: klass.instance_eval { def method } creates klass::method
         | 
| 144 | 
            -
                    #but klass.class_eval { def method } creates klass#method
         | 
| 145 183 |  | 
| 146 184 | 
             
                    command_class.instance_variable_set("@name", new_name)
         | 
| 147 185 | 
             
                    command_class.instance_variable_set("@parent", parent)
         | 
| @@ -152,8 +190,27 @@ module Command | |
| 152 190 | 
             
                    return command_class
         | 
| 153 191 | 
             
                  end
         | 
| 154 192 |  | 
| 193 | 
            +
                  def each_command(&block)
         | 
| 194 | 
            +
                    block.call(self)
         | 
| 195 | 
            +
                  end
         | 
| 196 | 
            +
             | 
| 197 | 
            +
                  def visit(terms, visitor)
         | 
| 198 | 
            +
                    result = visitor.arrive_at(terms, self)
         | 
| 199 | 
            +
             | 
| 200 | 
            +
                    if(terms.empty?)
         | 
| 201 | 
            +
                      return visitor.command_out_of_terms(self)
         | 
| 202 | 
            +
                    else
         | 
| 203 | 
            +
                      return visitor.extra_terms(terms, self)
         | 
| 204 | 
            +
                    end
         | 
| 205 | 
            +
                  end
         | 
| 206 | 
            +
             | 
| 207 | 
            +
                  def single_consume(argument, arg_hash, subject, terms)
         | 
| 208 | 
            +
                    arg_hash.merge! argument.consume(subject, terms)
         | 
| 209 | 
            +
                  end
         | 
| 210 | 
            +
             | 
| 155 211 | 
             
                  def defined
         | 
| 156 212 | 
             
                    @defined = true
         | 
| 213 | 
            +
                    create_argument_methods
         | 
| 157 214 | 
             
                  end
         | 
| 158 215 |  | 
| 159 216 | 
             
                  def defined?
         | 
| @@ -162,18 +219,8 @@ module Command | |
| 162 219 |  | 
| 163 220 | 
             
                  def inspect
         | 
| 164 221 | 
             
                    arguments = argument_list.map{|arg| arg.name}
         | 
| 165 | 
            -
                    argument_list.each do |argument|
         | 
| 166 | 
            -
                      arguments << argument.name
         | 
| 167 | 
            -
                    end
         | 
| 168 | 
            -
                    return "#<Class:#{self.object_id} - Command:#{path().join(" ")}(#{arguments.join(", ")})>"
         | 
| 169 | 
            -
                  end
         | 
| 170 222 |  | 
| 171 | 
            -
             | 
| 172 | 
            -
                    if @parent.nil?
         | 
| 173 | 
            -
                      [@name]
         | 
| 174 | 
            -
                    else
         | 
| 175 | 
            -
                      @parent.path + [@name]
         | 
| 176 | 
            -
                    end
         | 
| 223 | 
            +
                    return "#<Class:#{"%#x" % self.object_id} - Command:#{path().join(" ")}(#{arguments.join(", ")})>"
         | 
| 177 224 | 
             
                  end
         | 
| 178 225 |  | 
| 179 226 | 
             
                  def documentation(width, parent="")
         | 
| @@ -186,21 +233,22 @@ module Command | |
| 186 233 | 
             
                  end
         | 
| 187 234 |  | 
| 188 235 | 
             
                  def short_docs(width, parent="")
         | 
| 189 | 
            -
                    docs =  | 
| 190 | 
            -
             | 
| 191 | 
            -
             | 
| 192 | 
            -
             | 
| 193 | 
            -
             | 
| 194 | 
            -
             | 
| 195 | 
            -
             | 
| 196 | 
            -
             | 
| 236 | 
            +
                    docs = [path]
         | 
| 237 | 
            +
                    docs += argument_list.map do |arg|
         | 
| 238 | 
            +
                      if arg.required?
         | 
| 239 | 
            +
                        "<#{arg.name}>"
         | 
| 240 | 
            +
                      else
         | 
| 241 | 
            +
                        "[#{arg.name}]"
         | 
| 242 | 
            +
                      end
         | 
| 243 | 
            +
                    end
         | 
| 244 | 
            +
                    
         | 
| 197 245 |  | 
| 198 | 
            -
             | 
| 199 | 
            -
                    subject.required_fields(*(@subject_requirements.uniq))
         | 
| 246 | 
            +
                    return docs.join(" ").wrap(width)
         | 
| 200 247 | 
             
                  end
         | 
| 201 248 |  | 
| 202 249 | 
             
                  def completion_list(terms, prefix, subject)
         | 
| 203 250 | 
             
                    arguments = argument_list.dup
         | 
| 251 | 
            +
                    subject = subject.get_image(subject_requirements)
         | 
| 204 252 |  | 
| 205 253 | 
             
                    until(terms.empty? or arguments.empty?) do
         | 
| 206 254 | 
             
                      arguments.first.match_terms(subject, terms, arguments)
         | 
| @@ -214,42 +262,28 @@ module Command | |
| 214 262 | 
             
                    return completion
         | 
| 215 263 | 
             
                  end
         | 
| 216 264 |  | 
| 217 | 
            -
                  def create_argument_methods | 
| 218 | 
            -
                     | 
| 219 | 
            -
                       | 
| 265 | 
            +
                  def create_argument_methods
         | 
| 266 | 
            +
                    names = (argument_list + parent_arguments).inject([]) do |list, arg|
         | 
| 267 | 
            +
                      list + [*arg.names]
         | 
| 220 268 | 
             
                    end
         | 
| 221 | 
            -
                    private(name)
         | 
| 222 | 
            -
             | 
| 223 | 
            -
                    define_method(name.to_s + "=") do |value|
         | 
| 224 | 
            -
                      @arg_hash[name] = value
         | 
| 225 | 
            -
                    end
         | 
| 226 | 
            -
                    private(name.to_s + "=")
         | 
| 227 | 
            -
                  end
         | 
| 228 | 
            -
             | 
| 229 | 
            -
                  def embed_argument(arg)
         | 
| 230 | 
            -
                    names = [*arg.name]
         | 
| 231 269 |  | 
| 232 270 | 
             
                    names.each do |name|
         | 
| 233 | 
            -
                       | 
| 234 | 
            -
             | 
| 271 | 
            +
                      define_method(name) do
         | 
| 272 | 
            +
                        @arg_hash[name]
         | 
| 273 | 
            +
                      end
         | 
| 274 | 
            +
                      private(name)
         | 
| 235 275 |  | 
| 236 | 
            -
             | 
| 237 | 
            -
             | 
| 238 | 
            -
                        raise NoMethodError, "Can't define required arguments after optionals"
         | 
| 276 | 
            +
                      define_method(name.to_s + "=") do |value|
         | 
| 277 | 
            +
                        @arg_hash[name] = value
         | 
| 239 278 | 
             
                      end
         | 
| 279 | 
            +
                      private(name.to_s + "=")
         | 
| 240 280 | 
             
                    end
         | 
| 241 | 
            -
             | 
| 242 | 
            -
                    argument_list << arg
         | 
| 243 | 
            -
                  end
         | 
| 244 | 
            -
             | 
| 245 | 
            -
                  def optional_argument(arg, values=nil, &get_values)
         | 
| 246 | 
            -
                    optional.argument(arg, values, &get_values)
         | 
| 247 281 | 
             
                  end
         | 
| 248 | 
            -
             | 
| 249 282 | 
             
                end
         | 
| 250 283 |  | 
| 251 284 | 
             
                extend DSL::CommandDefinition
         | 
| 252 | 
            -
                extend  | 
| 285 | 
            +
                extend CommandClassMethods
         | 
| 286 | 
            +
                extend Common
         | 
| 253 287 | 
             
                include DSL::Action
         | 
| 254 288 |  | 
| 255 289 | 
             
                class DontResume; end
         | 
| @@ -287,7 +321,7 @@ module Command | |
| 287 321 | 
             
                end
         | 
| 288 322 |  | 
| 289 323 | 
             
                def all_arguments
         | 
| 290 | 
            -
                  self.class.argument_list
         | 
| 324 | 
            +
                  self.class.argument_list + self.class.parent_arguments
         | 
| 291 325 | 
             
                end
         | 
| 292 326 |  | 
| 293 327 | 
             
                def subject_requirements
         | 
| @@ -296,7 +330,7 @@ module Command | |
| 296 330 |  | 
| 297 331 | 
             
                def inspect
         | 
| 298 332 | 
             
                  name = self.class.name
         | 
| 299 | 
            -
                  return "#<Command:#{name}>:#{self.object_id} #{@arg_hash. | 
| 333 | 
            +
                  return "#<Command:#{name}>:#{"%#x" % self.object_id} #{@arg_hash.inspect}"
         | 
| 300 334 | 
             
                end
         | 
| 301 335 |  | 
| 302 336 | 
             
                def parent
         | 
| @@ -306,7 +340,7 @@ module Command | |
| 306 340 | 
             
                def go(collector)
         | 
| 307 341 | 
             
                  return if @resume_from == DontResume
         | 
| 308 342 | 
             
                  unless self.respond_to? :execute
         | 
| 309 | 
            -
                    raise CommandException, "#{ | 
| 343 | 
            +
                    raise CommandException, "#{self.class.path}: command declared but no action defined" 
         | 
| 310 344 | 
             
                  end
         | 
| 311 345 |  | 
| 312 346 | 
             
                  @main_collector = collector
         | 
| @@ -387,34 +421,6 @@ module Command | |
| 387 421 | 
             
                  @validation_problem = nil
         | 
| 388 422 | 
             
                end
         | 
| 389 423 |  | 
| 390 | 
            -
                def yield_consumer(argument, &block)
         | 
| 391 | 
            -
                  blk = proc(&block)
         | 
| 392 | 
            -
                  consumer = proc do |args|
         | 
| 393 | 
            -
                    raise TypeError, "#{args.inspect} isn't an Array!" unless Array === args
         | 
| 394 | 
            -
                    begin
         | 
| 395 | 
            -
                      if args.empty?
         | 
| 396 | 
            -
                        raise OutOfArgumentsException, "argument #{argument.inspect} required!"
         | 
| 397 | 
            -
                      end
         | 
| 398 | 
            -
                      @arg_hash.merge! argument.consume(@subject, args)
         | 
| 399 | 
            -
                      return args
         | 
| 400 | 
            -
                    rescue ArgumentInvalidException => aie
         | 
| 401 | 
            -
                      @validation_problem = aie
         | 
| 402 | 
            -
                      raise
         | 
| 403 | 
            -
                    end
         | 
| 404 | 
            -
                  end
         | 
| 405 | 
            -
                  blk[consumer]
         | 
| 406 | 
            -
                end
         | 
| 407 | 
            -
             | 
| 408 | 
            -
                def each_consumer(&block)
         | 
| 409 | 
            -
                  begin
         | 
| 410 | 
            -
                    all_arguments.each do |argument|
         | 
| 411 | 
            -
                      yield_consumer(argument, &block)
         | 
| 412 | 
            -
                    end
         | 
| 413 | 
            -
                  rescue OutOfArgumentsException
         | 
| 414 | 
            -
                  end
         | 
| 415 | 
            -
                  @validation_problem = nil
         | 
| 416 | 
            -
                end
         | 
| 417 | 
            -
             | 
| 418 424 | 
             
                def undoable?
         | 
| 419 425 | 
             
                  return false
         | 
| 420 426 | 
             
                end
         |