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