module CLIUtils # Pref Class # An individual preference class Pref include Messaging # Constant defining an Action ASSET_TYPE_ACTION = 0 # Constant defining a Behavior ASSET_TYPE_BEHAVIOR = 1 # Constant defining a Validator ASSET_TYPE_VALIDATOR = 2 # Contains a listing of asset names for classes and # file suffixes. # @return [Array] @@asset_labels = [ { class_suffix: 'Action', file_suffix: 'action' }, { class_suffix: 'Behavior', file_suffix: 'behavior' }, { class_suffix: 'Validator', file_suffix: 'validator' } ] # Stores the answer to this Pref. # @return [String, Symbol] attr_accessor :answer # Stores instantiated Behavior objects. # @return [Array] attr_accessor :behavior_objects # Stores the behaviors that this Pref conforms to. # @return [Array] attr_accessor :behaviors # Stores a key to reference this pref in a Configurator. # @return [String, Symbol] attr_accessor :config_key # Stores a Configurator section to stick this Pref under. # @return [String, Symbol] attr_accessor :config_section # Stores the default text. # @return [String] attr_accessor :default # Stores the last error message. # @return [String] attr_accessor :last_error_message # Stores the valid options the user can pick from. # @return [Array] attr_accessor :options # Stores the message and behavior that should be # executed after the prompt is delivered. # @return [Hash] attr_accessor :post # Stores the message and behavior that should be # executed before the prompt is delivered. # @return [Hash] attr_accessor :pre # Stores the prereqs information. # @return [Array] attr_accessor :prereqs # Stores the prompt text. # @return [String] attr_accessor :prompt_text # Stores instantiated Validators # @return [Array] attr_accessor :validator_objects # Stores key/value combinations required to show this Pref. # @return [Hash] attr_accessor :validators # Initializes a new Pref via passed-in parameters. Also # initializes objects for each Validator and Behavior on # this Pref. # @param [Hash] params Parameters to initialize # @return [void] def initialize(params = {}) @behavior_objects = [] @validator_objects = [] # Assign all of the passed params as instance variables. params.each { |key, value| send("#{ key }=", value) } # Instantiate any listed Behaviors or Validators. @behaviors.each { |b| _init_and_add_behavior(b) } if @behaviors @validators.each { |v| _init_and_add_validator(v) } if @validators end # Custom equality operator for this class. # @param [Pref] other # @return [Boolean] def ==(other) @config_key == other.config_key && @config_section == other.config_section && @prompt_text == other.prompt_text end # Delivers the prompt the user. Handles retries # after incorrect answers, validation, behavior # evaluation, and pre-/post-behaviors. # @param [String] default The default option # @return [void] def deliver(default = nil) # Design decision: the pre-prompt behavior # gets evaluated *once*, not every time the # user gets prompted. This prevents multiple # evaluations when bad options are provided. _eval_pre if @pre valid_option_chosen = false until valid_option_chosen response = prompt(@prompt_text, default) if validate(response) valid_option_chosen = true @answer = evaluate_behaviors(response) _eval_post if @post else messenger.error(@last_error_message) end end end # Runs the passed text through this Pref's behaviors. # @param [String] text The text to evaluate # @return [String] def evaluate_behaviors(text) modified_text = text if @behavior_objects @behavior_objects.each do |b| modified_text = b.evaluate(modified_text) end end modified_text end # Validates a text against this pref's options and # validators. # @param [String] text The text to validate # @return [Boolean] def validate(text) _check_options(text) && _check_validators(text) end private # Validates a text against the options for this Pref # @param [String] text The text to validate # @return [Boolean] def _check_options(text) ret = true if @options unless @options.include?(text) @last_error_message = "Invalid option chosen (\"#{ text }\"); " \ "valid options are: #{ options }" ret = false end end ret end # Validates a text against the validators for this Pref # @param [String] text The text to validate # @return [Boolean] def _check_validators(text) ret = true if @validator_objects @validator_objects.each do |v| v.validate(text) unless v.is_valid @last_error_message = v.message ret = false end end end ret end # Evaluates the pre-prompt Hash and does the right thing. :) # @return [void] def _eval_pre info(@pre[:message]) prompt('Press enter to continue') if (@pre[:action]) action_obj = _init_action(@pre[:action][:name]) action_obj.run if action_obj end end # Evaluates the post-prompt Hash and does the right thing. :) # @return [void] def _eval_post info(@post[:message]) prompt('Press enter to continue') if (@post[:action]) action_obj = _init_action(@post[:action][:name]) action_obj.run if action_obj end end # Attempts to instantiate a Pre or Post Action based on name; if # successful, the new object gets placed in @validator_objects # @param [String] path_or_name The path to or name of the Action # @return [void] def _init_action(path_or_name) obj = _load_asset(ASSET_TYPE_ACTION, path_or_name) unless obj.nil? obj.parameters = @pre[:action][:parameters] end obj end # Attempts to instantiate a Behavior based on name; if # successful, the new object gets placed in @behavior_objects # @param [Hash] behavior_hash The Behavior attributes # @return [void] def _init_and_add_behavior(behavior_hash) obj = _load_asset(ASSET_TYPE_BEHAVIOR, behavior_hash[:name]) unless obj.nil? obj.parameters = behavior_hash[:parameters] @behavior_objects << obj end end # Attempts to instantiate a Validator based on name; if # successful, the new object gets placed in @validator_objects # @param [String] path_or_name The path to or name of the Validator # @return [void] def _init_and_add_validator(path_or_name) obj = _load_asset(ASSET_TYPE_VALIDATOR, path_or_name) unless obj.nil? @validator_objects << obj end end # General function to load an asset (a Validator or Behavior). # Note that if an asset isn't found, the user is warned, but # execution continues. # @param [Integer] type ASSET_TYPE_BEHAVIOR or ASSET_TYPE_VALIDATOR # @param [String] path_or_name The path to or name of the asset # @return [Object] def _load_asset(type, path_or_name) asset_found = false if File.file?(File.expand_path(path_or_name)) # If the file exists, we're assuming that the user # passed a filepath. asset_found = true asset_path = File.expand_path(path_or_name) if path_or_name.start_with?('~') asset_name = File.basename(path_or_name, '.*').camelize end unless asset_found # If the file doesn't exist, look to see if it's been # pre-registered. symbol = File.basename(path_or_name, '.*').camelize.to_sym case type when Pref::ASSET_TYPE_ACTION if CLIUtils::Prefs.registered_actions.key?(symbol) asset_found = true asset_path = CLIUtils::Prefs.registered_actions[symbol][:path] asset_name = CLIUtils::Prefs.registered_actions[symbol][:class] end when Pref::ASSET_TYPE_BEHAVIOR if CLIUtils::Prefs.registered_behaviors.key?(symbol) asset_found = true asset_path = CLIUtils::Prefs.registered_behaviors[symbol][:path] rescue '' asset_name = CLIUtils::Prefs.registered_behaviors[symbol][:class] rescue '' end when Pref::ASSET_TYPE_VALIDATOR if CLIUtils::Prefs.registered_validators.key?(symbol) asset_found = true asset_path = CLIUtils::Prefs.registered_validators[symbol][:path] rescue '' asset_name = CLIUtils::Prefs.registered_validators[symbol][:class] rescue '' end end end unless asset_found # If the file doesn't exist and there's no pre-registered # asset, as a last check, look for it as a built-in asset. _default = File.join(File.dirname(__FILE__), "pref_#{ @@asset_labels[type][:file_suffix] }s") asset_path = File.join(_default, "#{ path_or_name }_#{ @@asset_labels[type][:file_suffix] }") asset_name = "#{ path_or_name.camelize }#{ @@asset_labels[type][:class_suffix] }" end # Try to load and instantiate the asset. If that fails, warn # the user with a message and skip over it. begin require File.expand_path(asset_path) Object.const_get('CLIUtils').const_get(asset_name).new rescue LoadError => e messenger.warn("Skipping undefined Pref #{ @@asset_labels[type][:class_suffix] }: #{ path_or_name }") nil end end end end