mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
69742ca8fa
Conflicts: activemodel/lib/active_model/core.rb activemodel/test/cases/state_machine/event_test.rb activemodel/test/cases/state_machine/state_transition_test.rb activerecord/lib/active_record/validations.rb activerecord/test/cases/validations/i18n_validation_test.rb activeresource/lib/active_resource.rb activeresource/test/abstract_unit.rb
87 lines
3.1 KiB
Ruby
87 lines
3.1 KiB
Ruby
require 'active_resource/exceptions'
|
|
require 'active_support/core_ext/array/wrap'
|
|
|
|
module ActiveResource
|
|
class ResourceInvalid < ClientError #:nodoc:
|
|
end
|
|
|
|
# Active Resource validation is reported to and from this object, which is used by Base#save
|
|
# to determine whether the object in a valid state to be saved. See usage example in Validations.
|
|
class Errors < ActiveModel::Errors
|
|
# Grabs errors from the XML response.
|
|
def from_xml(xml)
|
|
clear
|
|
humanized_attributes = @base.attributes.keys.inject({}) { |h, attr_name| h.update(attr_name.humanize => attr_name) }
|
|
messages = Array.wrap(Hash.from_xml(xml)['errors']['error']) rescue []
|
|
messages.each do |message|
|
|
attr_message = humanized_attributes.keys.detect do |attr_name|
|
|
if message[0, attr_name.size + 1] == "#{attr_name} "
|
|
add humanized_attributes[attr_name], message[(attr_name.size + 1)..-1]
|
|
end
|
|
end
|
|
|
|
self[:base] << message if attr_message.nil?
|
|
end
|
|
end
|
|
end
|
|
|
|
# Module to support validation and errors with Active Resource objects. The module overrides
|
|
# Base#save to rescue ActiveResource::ResourceInvalid exceptions and parse the errors returned
|
|
# in the web service response. The module also adds an +errors+ collection that mimics the interface
|
|
# of the errors provided by ActiveRecord::Errors.
|
|
#
|
|
# ==== Example
|
|
#
|
|
# Consider a Person resource on the server requiring both a +first_name+ and a +last_name+ with a
|
|
# <tt>validates_presence_of :first_name, :last_name</tt> declaration in the model:
|
|
#
|
|
# person = Person.new(:first_name => "Jim", :last_name => "")
|
|
# person.save # => false (server returns an HTTP 422 status code and errors)
|
|
# person.valid? # => false
|
|
# person.errors.empty? # => false
|
|
# person.errors.count # => 1
|
|
# person.errors.full_messages # => ["Last name can't be empty"]
|
|
# person.errors[:last_name] # => ["can't be empty"]
|
|
# person.last_name = "Halpert"
|
|
# person.save # => true (and person is now saved to the remote service)
|
|
#
|
|
module Validations
|
|
def self.included(base) # :nodoc:
|
|
base.class_eval do
|
|
alias_method_chain :save, :validation
|
|
end
|
|
end
|
|
|
|
# Validate a resource and save (POST) it to the remote web service.
|
|
def save_with_validation
|
|
save_without_validation
|
|
true
|
|
rescue ResourceInvalid => error
|
|
errors.from_xml(error.response.body)
|
|
false
|
|
end
|
|
|
|
# Checks for errors on an object (i.e., is resource.errors empty?).
|
|
#
|
|
# ==== Examples
|
|
# my_person = Person.create(params[:person])
|
|
# my_person.valid?
|
|
# # => true
|
|
#
|
|
# my_person.errors.add('login', 'can not be empty') if my_person.login == ''
|
|
# my_person.valid?
|
|
# # => false
|
|
def valid?
|
|
errors.empty?
|
|
end
|
|
|
|
# Returns the Errors object that holds all information about attribute error messages.
|
|
def errors
|
|
@errors ||= Errors.new(self)
|
|
end
|
|
end
|
|
|
|
class Base
|
|
include Validations
|
|
end
|
|
end
|