require 'volt/models/errors' require 'volt/models/validators/format_validator' require 'volt/models/validators/email_validator' require 'volt/models/validators/length_validator' require 'volt/models/validators/numericality_validator' require 'volt/models/validators/phone_number_validator' require 'volt/models/validators/presence_validator' require 'volt/models/validators/unique_validator' module Volt # Include in any class to get validation logic module Validations module ClassMethods def validate(field_name = nil, options = nil, &block) if block if field_name || options fail 'validate should be passed a field name and options or a block, not both.' end self.custom_validations ||= [] custom_validations << block else self.validations ||= {} validations[field_name] = options end end end def self.included(base) base.send :extend, ClassMethods base.class_attribute(:custom_validations, :validations) end # Once a field is ready, we can use include_in_errors! to start # showing its errors. def mark_field!(field_name) marked_fields[field_name] = true end def marked_fields @marked_fields ||= ReactiveHash.new end # Marks all fields, useful for when a model saves. def mark_all_fields! validations = self.class.validations if validations validations.each_key do |key| mark_field!(key.to_sym) end end end def marked_errors errors(true) end # server errors are errors that come back from the server when we save! # Any changes to the associated fields will clear the error until another # save! def server_errors @server_errors ||= ReactiveHash.new end # When a field is changed, we want to clear any errors from the server def clear_server_errors(key) @server_errors.delete(key) end def errors(marked_only=false) @errors ||= Errors.new if marked_only # Only return the fields that have been marked @errors.to_h.select {|key,_| marked_fields[key] } else @errors end end # TODO: Errors is being called for any validation change. We should have errors return a # hash like object that only calls the validation for each one. def validate! errors.clear run_validations.then do # See if any server errors are in place and merge them in if they are if Volt.client? errors.merge!(server_errors.to_h) end end.then do run_custom_validations end.then do # Return the errors object errors end end # Returns true if any of the changed fields now has an error # @return [Boolean] true if one of the changed fields has an error. def error_in_changed_attributes? errs = errors changed_attributes.each_pair do |key, _| # If any of the fields with errors are also the ones that were return true if errs[key] end return false end private # Runs through each of the normal validations. # @return [Promise] a promsie to run all validations def run_validations promise = Promise.new.resolve(nil) validations = self.class.validations if validations # Run through each validation validations.each_pair do |field_name, options| options.each_pair do |validation, args| # Call the specific validator, then merge the results back # into one large errors hash. klass = validation_class(validation, args) if klass # Chain on the promises promise = promise.then do klass.validate(self, field_name, args) end.then do |errs| errors.merge!(errs) end else fail "validation type #{validation} is not specified." end end end end return promise end def run_custom_validations promise = Promise.new.resolve(nil) # Call all of the custom validations custom_validations = self.class.custom_validations if custom_validations custom_validations.each do |custom_validation| # Add to the promise chain promise = promise.then do # Run the validator in the context of the model instance_exec(&custom_validation) end.then do |errs| errors.merge!(errs) end end end return promise end def validation_class(validation, args) Volt.const_get(:"#{validation.camelize}Validator") rescue NameError => e puts "Unable to find #{validation} validator" end end end