serialization.rb 4.3 KB
Newer Older
1 2 3
require 'active_support/core_ext/hash/except'
require 'active_support/core_ext/hash/slice'

4
module ActiveModel
5 6 7 8 9 10 11 12 13 14 15 16
  # == Active Model Serialization
  #
  # Provides a basic serialization to a serializable_hash for your object.
  #
  # A minimal implementation could be:
  #
  #   class Person
  #     include ActiveModel::Serialization
  #
  #     attr_accessor :name
  #
  #     def attributes
17
  #       {'name' => nil}
18 19 20 21 22 23 24 25 26 27
  #     end
  #   end
  #
  # Which would provide you with:
  #
  #   person = Person.new
  #   person.serializable_hash   # => {"name"=>nil}
  #   person.name = "Bob"
  #   person.serializable_hash   # => {"name"=>"Bob"}
  #
28 29 30 31 32
  # You need to declare an attributes hash which contains the attributes
  # you want to serialize. When called, serializable hash will use
  # instance methods that match the name of the attributes hash's keys.
  # In order to override this behavior, take a look at the private
  # method read_attribute_for_serialization.
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
  #
  # Most of the time though, you will want to include the JSON or XML
  # serializations. Both of these modules automatically include the
  # ActiveModel::Serialization module, so there is no need to explicitly
  # include it.
  #
  # So a minimal implementation including XML and JSON would be:
  #
  #   class Person
  #     include ActiveModel::Serializers::JSON
  #     include ActiveModel::Serializers::Xml
  #
  #     attr_accessor :name
  #
  #     def attributes
48
  #       {'name' => nil}
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66
  #     end
  #   end
  #
  # Which would provide you with:
  #
  #   person = Person.new
  #   person.serializable_hash   # => {"name"=>nil}
  #   person.as_json             # => {"name"=>nil}
  #   person.to_json             # => "{\"name\":null}"
  #   person.to_xml              # => "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<serial-person...
  #
  #   person.name = "Bob"
  #   person.serializable_hash   # => {"name"=>"Bob"}
  #   person.as_json             # => {"name"=>"Bob"}
  #   person.to_json             # => "{\"name\":\"Bob\"}"
  #   person.to_xml              # => "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<serial-person...
  #
  # Valid options are <tt>:only</tt>, <tt>:except</tt> and <tt>:methods</tt> .
67
  module Serialization
68 69 70 71 72
    def serializable_hash(options = nil)
      options ||= {}

      attribute_names = attributes.keys.sort
      if only = options[:only]
73
        attribute_names &= Array(only).map(&:to_s)
74
      elsif except = options[:except]
75
        attribute_names -= Array(except).map(&:to_s)
76 77 78 79 80
      end

      hash = {}
      attribute_names.each { |n| hash[n] = read_attribute_for_serialization(n) }

81
      method_names = Array(options[:methods]).select { |n| respond_to?(n) }
82
      method_names.each { |n| hash[n.to_s] = send(n) }
83 84

      serializable_add_includes(options) do |association, records, opts|
85
        hash[association.to_s] = if records.is_a?(Enumerable)
86 87 88 89 90
          records.map { |a| a.serializable_hash(opts) }
        else
          records.serializable_hash(opts)
        end
      end
91

92
      hash
93
    end
94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122

    private

      # Hook method defining how an attribute value should be retrieved for
      # serialization. By default this is assumed to be an instance named after
      # the attribute. Override this method in subclasses should you need to
      # retrieve the value for a given attribute differently:
      #
      #   class MyClass
      #     include ActiveModel::Validations
      #
      #     def initialize(data = {})
      #       @data = data
      #     end
      #
      #     def read_attribute_for_serialization(key)
      #       @data[key]
      #     end
      #   end
      #
      alias :read_attribute_for_serialization :send

      # Add associations specified via the <tt>:include</tt> option.
      #
      # Expects a block that takes as arguments:
      #   +association+ - name of the association
      #   +records+     - the association record(s) to be serialized
      #   +opts+        - options for the association records
      def serializable_add_includes(options = {}) #:nodoc:
123
        return unless includes = options[:include]
124

125 126
        unless includes.is_a?(Hash)
          includes = Hash[Array(includes).map { |n| n.is_a?(Hash) ? n.to_a.first : [n, {}] }]
127 128
        end

129
        includes.each do |association, opts|
130 131 132 133 134
          if records = send(association)
            yield association, records, opts
          end
        end
      end
135
  end
136
end