gabynaiman/rasti-form

View on GitHub
lib/rasti/form.rb

Summary

Maintainability
A
0 mins
Test Coverage
require 'rasti-model'
require 'multi_require'

module Rasti
  class Form < Rasti::Model

    extend MultiRequire

    require_relative_pattern 'form/*'

    include Validable

    alias_method :__initialize__, :initialize
    private :__initialize__

    def initialize(attributes={})
      assign_attributes! attributes
      validate_type_casting!
      validate!
    end

    def assigned?(attr_name)
      assigned_attribute? attr_name.to_sym
    end

    private

    def assign_attributes!(attributes)
      __initialize__ attributes

    rescue Rasti::Model::UnexpectedAttributesError => ex
      ex.attributes.each do |attr_name|
        errors[attr_name] << 'unexpected attribute'
      end

    ensure
      raise_if_errors!
    end

    def validate_type_casting!
      cast_attributes!

    rescue Rasti::Types::CompoundError => ex
      ex.errors.each do |key, messages|
        errors[key] += messages
      end

    ensure
      raise_if_errors!
    end

    def assert_present(attr_name)
      if !errors.key?(attr_name)
        assert attr_name, assigned?(attr_name) && !public_send(attr_name).nil?, 'not present'
      end
    end

    def assert_not_present(attr_name)
      assert attr_name, !assigned?(attr_name) || public_send(attr_name).nil?, 'is present'
    end

    def assert_not_empty(attr_name)
      if assert_present attr_name
        value = public_send attr_name
        assert attr_name, value.is_a?(String) ? !value.strip.empty? : !value.empty?, 'is empty'
      end
    end

    def assert_included_in(attr_name, set)
      if assert_present attr_name
        assert attr_name, set.include?(public_send(attr_name)), "not included in #{set.map(&:inspect).join(', ')}"
      end
    end

    def assert_range(attr_name_from, attr_name_to)
      if assert_present(attr_name_from) && assert_present(attr_name_to)
        assert attr_name_from, public_send(attr_name_from) <= public_send(attr_name_to), 'invalid range'
      end
    end

  end
end