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

4
module ActiveModel
5
  # == Active \Model \Serialization
6
  #
7
  # Provides a basic serialization to a serializable_hash for your objects.
8 9 10 11 12 13 14 15 16
  #
  # 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 33
  # An +attributes+ hash must be defined and should contain any attributes you
  # need to be serialized. Attributes must be strings, not symbols.
  # 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+.
  #
34 35 36
  # ActiveModel::Serializers::JSON module automatically includes
  # the <tt>ActiveModel::Serialization</tt> module, so there is no need to
  # explicitly include <tt>ActiveModel::Serialization</tt>.
37
  #
38
  # A minimal implementation including JSON would be:
39 40 41 42 43 44 45
  #
  #   class Person
  #     include ActiveModel::Serializers::JSON
  #
  #     attr_accessor :name
  #
  #     def attributes
46
  #       {'name' => nil}
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61
  #     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.name = "Bob"
  #   person.serializable_hash   # => {"name"=>"Bob"}
  #   person.as_json             # => {"name"=>"Bob"}
  #   person.to_json             # => "{\"name\":\"Bob\"}"
  #
62 63
  # Valid options are <tt>:only</tt>, <tt>:except</tt>, <tt>:methods</tt> and
  # <tt>:include</tt>. The following are all valid examples:
64
  #
65 66 67
  #   person.serializable_hash(only: 'name')
  #   person.serializable_hash(include: :address)
  #   person.serializable_hash(include: { address: { only: 'city' }})
68
  module Serialization
69 70 71 72 73
    # Returns a serialized hash of your object.
    #
    #   class Person
    #     include ActiveModel::Serialization
    #
74
    #     attr_accessor :name, :age
75 76 77 78 79 80 81 82 83 84 85 86 87 88
    #
    #     def attributes
    #       {'name' => nil, 'age' => nil}
    #     end
    #
    #     def capitalized_name
    #       name.capitalize
    #     end
    #   end
    #
    #   person = Person.new
    #   person.name = 'bob'
    #   person.age  = 22
    #   person.serializable_hash                # => {"name"=>"bob", "age"=>22}
89
    #   person.serializable_hash(only: :name)   # => {"name"=>"bob"}
90 91 92
    #   person.serializable_hash(except: :name) # => {"age"=>22}
    #   person.serializable_hash(methods: :capitalized_name)
    #   # => {"name"=>"bob", "age"=>22, "capitalized_name"=>"Bob"}
93 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
    #
    # Example with <tt>:include</tt> option
    #
    #   class User
    #     include ActiveModel::Serializers::JSON
    #     attr_accessor :name, :notes # Emulate has_many :notes
    #     def attributes
    #       {'name' => nil}
    #     end
    #   end
    #
    #   class Note
    #     include ActiveModel::Serializers::JSON
    #     attr_accessor :title, :text
    #     def attributes
    #       {'title' => nil, 'text' => nil}
    #     end
    #   end
    #
    #   note = Note.new
    #   note.title = 'Battle of Austerlitz'
    #   note.text = 'Some text here'
    #
    #   user = User.new
    #   user.name = 'Napoleon'
    #   user.notes = [note]
    #
    #   user.serializable_hash
121
    #   # => {"name" => "Napoleon"}
122
    #   user.serializable_hash(include: { notes: { only: 'title' }})
123
    #   # => {"name" => "Napoleon", "notes" => [{"title"=>"Battle of Austerlitz"}]}
124 125 126
    def serializable_hash(options = nil)
      options ||= {}

T
Troy Kruthoff 已提交
127
      attribute_names = attributes.keys
128
      if only = options[:only]
129
        attribute_names &= Array(only).map(&:to_s)
130
      elsif except = options[:except]
131
        attribute_names -= Array(except).map(&:to_s)
132 133 134 135 136
      end

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

137
      Array(options[:methods]).each { |m| hash[m.to_s] = send(m) }
138 139

      serializable_add_includes(options) do |association, records, opts|
140 141
        hash[association.to_s] = if records.respond_to?(:to_ary)
          records.to_ary.map { |a| a.serializable_hash(opts) }
142 143 144 145
        else
          records.serializable_hash(opts)
        end
      end
146

147
      hash
148
    end
149 150 151 152 153 154 155 156 157

    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
158
      #     include ActiveModel::Serialization
159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176
      #
      #     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:
177
        return unless includes = options[:include]
178

179
        unless includes.is_a?(Hash)
180
          includes = Hash[Array(includes).flat_map { |n| n.is_a?(Hash) ? n.to_a : [[n, {}]] }]
181 182
        end

183
        includes.each do |association, opts|
184 185 186 187 188
          if records = send(association)
            yield association, records, opts
          end
        end
      end
189
  end
190
end