accessd/rails-settings-ui

View on GitHub
lib/rails-settings-ui/settings_form_coercible.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'dry-types'

module RailsSettingsUi
  module Types
    include Dry::Types.module

    module CustomCoercions

      class Symbol
        def self.[](value)
          value.to_sym
        end
      end

      class Hash
        def self.[](value)
          JSON.parse(value.gsub(/\=\>/, ':'))
        end
      end

      class Array
        def self.[](value)
          # array presented in checkboxes
          case value.class.name
          when 'Hash', 'ActiveSupport::HashWithIndifferentAccess'
            value.keys.map!(&:to_sym)
          when 'ActionController::Parameters'
            value.select{ |_,v| v == 'on' }.keys.map!(&:to_sym)
          else
            # or in select tag
            value.to_sym
          end
        end
      end

    end
  end

  class NotCoercibleError < StandardError;end

  class SettingsFormCoercible
    attr_reader :settings, :default_settings
    attr_accessor :coerced_settings

    COERCIONS_MAP = {
        String => Types::Coercible::String,
        Symbol => Types::CustomCoercions::Symbol,
        (1.class == Integer ? Integer : Fixnum) => Types::Params::Integer,
        ActiveSupport::HashWithIndifferentAccess => Types::CustomCoercions::Hash,
        ActiveSupport::Duration => Types::Params::Integer,
        Float => Types::Params::Float,
        Array => Types::CustomCoercions::Array,
        FalseClass => Types::Params::Bool,
        TrueClass => Types::Params::Bool
    }.freeze

    def initialize(default_settings, settings)
      @default_settings = default_settings
      @settings = settings
      @coerced_settings = {}
    end

    def coerce!
      settings.to_hash.symbolize_keys.each do |name, value|
        default_value_class = default_settings[name].class
        coercible_type = COERCIONS_MAP[default_value_class]
        raise NotCoercibleError, "can't coerce #{default_value_class}" unless coercible_type
        coerced_settings[name] = coercible_type[value]
      end
      set_default_boolean_value!(coerced_settings)
      coerced_settings
    end

    private

    def set_default_boolean_value!(settings)
      default_settings.each do |name, value|
        setting_value = settings[name.to_sym]
        next if setting_value.present?

        settings[name.to_sym] = case value.class.name
                                when 'TrueClass', 'FalseClass'
                                  false
                                when 'Array'
                                  []
                                end
      end
    end

  end
end