piotrmurach/tty-option

View on GitHub
lib/tty/option/parameter.rb

Summary

Maintainability
B
5 hrs
Test Coverage
# frozen_string_literal: true

require_relative "deep_dup"
require_relative "dsl/arity"
require_relative "dsl/conversion"

module TTY
  module Option
    class Parameter
      include Comparable
      include DSL::Arity
      include DSL::Conversion

      # Zero or more parameter arity pattern
      ZERO_OR_MORE_ARITY = /\*|any/.freeze

      # One or more parameter arity pattern
      ONE_OR_MORE_ARITY = /\+/.freeze

      # A parameter factory
      #
      # @api public
      def self.create(key, **settings, &block)
        new(key, **settings, &block)
      end

      # The key under which this parameter is registered
      #
      # @api public
      attr_reader :key

      # Create a parameter
      #
      # @param [Symbol] key
      #   the key to register this param under
      #
      # @api private
      def initialize(key, **settings, &block)
        @key = key
        @settings = {}
        settings.each do |name, val|
          case name.to_sym
          when :arity
            val = check_arity(val)
          when :default
            val = check_default(val)
          when :optional
            name, val = :required, check_required(!val)
          when :permit
            val = check_permitted(val)
          when :validate
            val = check_validation(val)
          end
          @settings[name.to_sym] = val
        end

        instance_eval(&block) if block_given?
      end

      def arity(value = (not_set = true))
        if not_set
          @settings.fetch(:arity) { default_arity }
        else
          @settings[:arity] = check_arity(value)
        end
      end

      def default_arity
        1
      end

      # Determine minimum boundary for arity parameter
      #
      # @api private
      def min_arity
        arity < 0 ? arity.abs - 1 : arity
      end

      # Check if multiple occurrences are allowed
      #
      # @return [Boolean]
      #
      # @api public
      def multiple?
        arity < 0 || 1 < arity.abs
      end

      def convert(value = (not_set = true))
        if not_set
          @settings[:convert]
        else
          @settings[:convert] = value
        end
      end

      def convert?
        @settings.key?(:convert) && !@settings[:convert].nil?
      end

      def default(value = (not_set = true))
        if not_set
          @settings[:default]
        else
          @settings[:default] = check_default(value)
        end
      end
      alias defaults default

      def default?
        @settings.key?(:default) && !@settings[:default].nil?
      end

      def desc(value = (not_set = true))
        if not_set
          @settings[:desc]
        else
          @settings[:desc] = value
        end
      end

      def desc?
        @settings.key?(:desc) && !@settings[:desc].nil?
      end

      # Check if this options is multi argument
      #
      # @api public
      def multi_argument?
        !convert.to_s.match(/list|array|map|hash|\w+s/).nil?
      end

      def optional
        @settings[:required] = false
      end

      def optional?
        !required?
      end

      def required
        @settings[:required] = check_required(true)
      end

      def required?
        @settings.fetch(:required, false)
      end

      def hidden
        @settings[:hidden] = true
      end

      def hidden?
        @settings.fetch(:hidden, false)
      end

      def display?
        desc? && !hidden?
      end

      def permit(value = (not_set = true))
        if not_set
          @settings[:permit]
        else
          @settings[:permit] = check_permitted(value)
        end
      end

      def permit?
        @settings.key?(:permit) && !@settings[:permit].nil?
      end

      def name(value = (not_set = true))
        if not_set
          @settings.fetch(:name) { default_name }
        else
          @settings[:name] = value
        end
      end

      def default_name
        key.to_s.tr("_", "-")
      end

      def validate(value = (not_set = true))
        if not_set
          @settings[:validate]
        else
          @settings[:validate] = check_validation(value)
        end
      end

      def validate?
        @settings.key?(:validate) && !@settings[:validate].nil?
      end

      def to_sym
        self.class.name.to_s.split(/::/).last.downcase.to_sym
      end

      # Compare this parameter name with the other
      #
      # @api public
      def <=>(other)
        name <=> other.name
      end

      # Compare parameters for equality based on type and name
      #
      # @api public
      def ==(other)
        return false unless instance_of?(other.class)

        name == other.name && to_h == other.to_h
      end

      # Compare parameters for equality based on type and name
      #
      # @api public
      def eql?(other)
        return false unless instance_of?(other.class)

        name.eql?(other.name) && to_h.eql?(other.to_h)
      end

      # Return a hash of this parameter settings
      #
      # @return [Hash] the names and values of this parameter
      #
      # @api public
      def to_h(&block)
        if block_given?
          @settings.each_with_object({}) do |(key, val), acc|
            k, v = *block.(key, val)
            acc[k] = v
          end
        else
          DeepDup.deep_dup(@settings)
        end
      end

      # Make a duplicate of this parameter
      #
      # @api public
      def dup
        super.tap do |param|
          param.instance_variable_set(:@key, DeepDup.deep_dup(@key))
          param.instance_variable_set(:@settings, DeepDup.deep_dup(@settings))
        end
      end

      private

      # @api private
      def check_arity(value)
        if value.nil?
          raise ConfigurationError,
                "#{to_sym} '#{name}' arity needs to be an Integer"
        end

        case value.to_s
        when ZERO_OR_MORE_ARITY then -1
        when ONE_OR_MORE_ARITY then -2
        else value.to_i
        end.tap do |val|
          if val.zero?
            raise ConfigurationError, "#{to_sym} '#{name}' arity cannot be zero"
          end
        end
      end

      # @api private
      def check_permitted(value)
        if value.respond_to?(:include?)
          value
        else
          raise ConfigurationError,
                "#{to_sym} '#{name}' permitted value needs to be " \
                "an Array or a Hash"
        end
      end

      def check_default(value)
        if !value.nil? && required?
          raise ConfigurationError,
                "#{to_sym} '#{name}' cannot have default value and be required"
        else
          value
        end
      end

      # @api private
      def check_required(value)
        if value && default?
          raise ConfigurationError,
                "#{to_sym} '#{name}' cannot be required and have default value"
        else
          value
        end
      end

      # @api private
      def check_validation(value)
        case value
        when NilClass
          raise ConfigurationError,
                "#{to_sym} '#{name}' validation needs to be a Proc or a Regexp"
        when Proc
          value
        when Regexp, String
          Regexp.new(value.to_s)
        else
          raise ConfigurationError,
                "#{to_sym} '#{name}' validation can only be a Proc or a Regexp"
        end
      end
    end # Parameter
  end # Option
end # TTY