piotrmurach/tty-option

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

Summary

Maintainability
A
45 mins
Test Coverage
# frozen_string_literal: true

module TTY
  module Option
    Error = Class.new(StandardError)

    # Raised when a parameter invariant is invalid
    ConfigurationError = Class.new(Error)

    # Raised when attempting to register already registered parameter
    ParameterConflict = Class.new(Error)

    # Raised when overriding already defined conversion
    ConversionAlreadyDefined = Class.new(Error)

    # Raised when conversion cannot be performed
    ConversionError = Class.new(Error)

    # Raised when conversion type isn't registered
    UnsupportedConversion = Class.new(Error)

    # Raised during command line input parsing
    class ParseError < Error
      attr_accessor :param

      # Format value
      #
      # @param [Object] value
      #   the value to format
      #
      # @example
      #   format_value([:a, 1])
      #   # => a:1
      #
      # @return [String]
      #
      # @api public
      def format_value(value)
        value.respond_to?(:to_ary) ? value.join(":") : value.to_s
      end
    end

    # Raised when found unrecognized parameter
    InvalidParameter = Class.new(ParseError)

    # Raised when an option matches more than one parameter option
    AmbiguousOption = Class.new(ParseError)

    # Raised when parameter argument doesn't match expected value
    class InvalidArgument < ParseError
      MESSAGE = "value of `%<value>s` fails validation for '%<name>s' %<type>s"

      def initialize(param_or_message, value = nil)
        if param_or_message.is_a?(Parameter)
          @param = param_or_message

          message = format(MESSAGE,
                           value: format_value(value),
                           name: param.name,
                           type: param.to_sym)
        else
          message = param_or_message
        end

        super(message)
      end
    end

    # Raised when number of parameter arguments doesn't match
    class InvalidArity < ParseError
      MESSAGE = "%<type>s '%<name>s' should appear %<expect>s but appeared %<actual>s"

      def initialize(param_or_message, arity = nil)
        if param_or_message.is_a?(Parameter)
          @param = param_or_message
          prefix = param.arity < 0 ? "at least " : ""
          expected_arity = param.arity < 0 ? param.arity.abs - 1 : param.arity

          message = format(MESSAGE,
                           type: param.to_sym,
                           name: param.name,
                           expect: prefix + pluralize("time", expected_arity),
                           actual: pluralize("time", arity))
        else
          message = param_or_message
        end

        super(message)
      end

      # Pluralize a noun
      #
      # @api private
      def pluralize(noun, count = 1)
        "#{count} #{noun}#{'s' unless count == 1}"
      end
    end

    # Raised when conversion provided with unexpected argument
    class InvalidConversionArgument < ParseError
      MESSAGE = "cannot convert value of `%<value>s` into '%<cast>s' type " \
                "for '%<name>s' %<type>s"

      def initialize(param, value)
        @param = param
        message = format(MESSAGE, value: value, cast: param.convert,
                          name: param.name, type: param.to_sym)
        super(message)
      end
    end

    # Raised when option requires an argument
    class MissingArgument < ParseError
      MESSAGE = "%<type>s %<name>s requires an argument"

      def initialize(param)
        @param = param
        message = format(MESSAGE, type: param.to_sym, name: param.name)
        super(message)
      end
    end

    # Raised when a parameter is required but not present
    class MissingParameter < ParseError
      MESSAGE = "%<type>s '%<name>s' must be provided"

      def initialize(param_or_message)
        if param_or_message.is_a?(Parameter)
          @param = param_or_message
          message = format(MESSAGE, name: param.name, type: param.to_sym)
        else
          message = param_or_message
        end

        super(message)
      end
    end

    # Raised when argument value isn't permitted
    class UnpermittedArgument < ParseError
      MESSAGE = "unpermitted value `%<value>s` for '%<name>s' %<type>s: " \
                "choose from %<choices>s"

      def initialize(param_or_message, value = nil)
        if param_or_message.is_a?(Parameter)
          @param = param_or_message
          message = format(MESSAGE,
                           value: format_value(value),
                           name: param.name,
                           type: param.to_sym,
                           choices: format_choices(param.permit))
        else
          message = param_or_message
        end

        super(message)
      end

      private

      # Format permitted choices
      #
      # @param [Array<Object>] choices
      #   the choices to format
      #
      # @return [String]
      #
      # @api private
      def format_choices(choices)
        choices.map { |val| format_value(val) }.join(", ")
      end
    end
  end # Option
end # TTY