jcamenisch/ENV_BANG

View on GitHub
lib/env_bang/classes.rb

Summary

Maintainability
A
1 hr
Test Coverage
class ENV_BANG
  module Classes
    class << self
      attr_writer :default_class

      def default_class
        @default_class ||= :StringUnlessFalsey
      end

      def cast(value, options = {})
        public_send(:"#{options.fetch(:class, default_class)}", value, options)
      end

      def boolean(value, options)
        !(value =~ /^(|0|disabled?|false|no|off)$/i)
      end

      def Array(value, options)
        item_options = options.merge(class: options.fetch(:of, default_class))
        value.split(',').map { |v| cast(v.strip, item_options) }
      end

      def Hash(value, options)
        key_options   = options.merge(class: options.fetch(:keys, Symbol))
        value_options = options.merge(class: options.fetch(:of, default_class))
        sep = options.fetch(:sep, ',')
        val_sep = options.fetch(:val_sep, ':')

        {}.tap do |h|
          value.split(sep).each do |pair|
            key, value = pair.split(val_sep, 2)
            h[cast(key.to_s.strip, key_options)] = cast(value.to_s.strip, value_options)
          end
        end
      end

      def Symbol(value, options)
        value.to_sym
      end

      def StringUnlessFalsey(value, options)
        boolean(value, options) && value
      end

      def Integer(value, options)
        Kernel.Integer(value)
      end

      def Float(value, options)
        Kernel.Float(value)
      end

      def String(value, options)
        Kernel.String(value)
      end

      def Rational(value, options)
        Kernel.Rational(value)
      end

      def Complex(value, options)
        Kernel.Complex(value)
      end

      def Pathname(value, options)
        Kernel.Pathname(value)
      end

      def URI(value, options)
        Kernel.URI(value)
      end

      def Date(value, options)
        Date.parse(value)
      end

      def DateTime(value, options)
        DateTime.parse(value)
      end

      def Time(value, options)
        Time.parse(value)
      end

      def Range(value, options = {})
        beginning, ending = value.split('...')
        if beginning && ending
          options[:exclusive] = true unless options.has_key?(:exclusive)
        else
          beginning, ending = value.split('..')
        end
        unless beginning && ending
          raise ArgumentError.new("Range '#{value}' cannot be parsed as a range. Must be in the form <start>..<end> or <start>...<end>")
        end

        options[:of] ||= Integer
        beginning = cast(beginning, class: options[:of])
        ending    = cast(ending, class: options[:of])

        if options[:exclusive]
          beginning...ending
        else
          beginning..ending
        end
      end

      def Regexp(value, options)
        Regexp.new(value)
      end
    end
  end
end